Assembly
Assembly
Assembly
University of Guadalajara
Information Sistems General Coordination.
Culture and Entertainment Web
*****************************************************************************
* Este tutorial foi traduzido para o Português por Jeferson Amaral. *
* e-mail: amaral@inf.ufsm.br *
*****************************************************************************
(1) any copy or republication of the entire document must show University of
Guadalajara as the source, and must include this notice; and
(2) any other use of this material must reference this manual and University
of Guadalajara, and the fact that the material is copyright by Hugo Perez
and is used by permission.
****************************************************************************
T U T O R I A L D E L I N G U A G E M A S S E M B L Y
--------------- --- ----------------- ---------------
Conteúdo:
1.Introdução
2.Conceitos Básicos
3.Programação Assembly
4.Instruções Assembly
5.Interrupções e gerência de arquivos
6.Macros e procedimentos
7.Exemplos de programas
8.Bibliografia
*****************************************************************************
CAPÍTULO 1: INTRODUÇÃO
Conteúdo:
--------------- // ---------------
1.2.Apresentação:
Este tutorial destina-se àquelas pessoas que nunca tiveram contato com a
Linguagem Assembly.
A segunda razão é que você pode ter um controle total sobre o PC ao fazer
uso do assembler.
Uma outra razão é que programas assembly são mais rápidos, menores e mais
poderosos do que os criados com outras linguagens.
Nosso intuito é oferecer um modo simples para que você consiga aprender
Assembly por si mesmo. Por tanto, qualquer comentário ou sugestão será
bem-vinda.
*****************************************************************************
--------------- // ---------------
Conteúdo:
2.1.1.Processador Central
2.1.2.Memória Principal
2.1.3.Unidades de Entrada e Saída
2.1.4.Unidades de Memória Auxiliar
Sistema Computacional.
2.1.1.Processador Central.
É também conhecido por CPU ou Unidade Central de Processamento, que por sua
vez é composta pela unidade de controle e unidade de lógica e aritmética.
Sua função consiste na leitura e escrita do conteúdo das células de memória,
regular o tráfego de dados entre as células de memória e registradores
especiais, e decodificar e executar as instruções de um programa.
2.1.2.Memória Principal.
Cada uma das células pode conter um valor numérico e é capaz de ser
endereçada, isto é, pode ser identificada de forma singular em relação às
outras células pelo uso de um número ou endereço.
Para que o computador possa ser útil para nós se faz necessário que o
processador se comunique com o exterior através de interfaces que permitem a
entrada e a saída de informação entre ele e a memória. Através do uso destas
comunicações é possível introduzir informação a ser processada e mais tarde
visualizar os dados processados.
Algumas das mais comuns unidades de entrada são o teclado e o mouse. As mais
comuns unidades de saída são a tela do monitor e a impressora.
Página 4
assembly
Estes e outros inconvenientes deram lugar às unidades de memória auxiliar,
periféricos. As mais comuns são as fitas e os discos magnéticos.
--------------- // ---------------
2.2.CONCEITOS BÁSICOS
Conteúdo:
2.2.1.Informação no computador:
2.2.1.1.Unidades de informação
2.2.1.2.Sistemas numéricos
2.2.1.3.Convertendo números binários para decimais
2.2.1.4.Convertendo números decimais para binários
2.2.1.5.Sistema hexadecimal
2.2.1.1.Unidades de informação
Um grupo de 16 bits é conhecido como palavra; uma palavra pode ser dividida
em grupos de 8 bits chamados bytes, e grupos de 4 bits chamados nibbles.
2.2.1.2.Sistemas numéricos
O sistema numérico que nós usamos diariamente é o decimal, mas este sistema
não é conveniente para máquinas, pois ali as informações têm que ser
codificadas de modo a interpretar os estados da corrente (ligado-desligado);
este modo de código faz com que tenhamos que conhecer o cálculo posicional
Página 5
assembly
que nos permitirá expressar um número em qualquer base onde precisarmos
dele.
Binary: 1 1 0 0 1
= 1 + 2 + 0 + 0 + 16 = 19 decimal.
2.2.1.5.Sistema hexadecimal
10;1011
0010;1011
0010=2;1011=11
Entretanto, observa-se que não podemos representar este número como 211,
isto seria um erro, uma vez que os números em hexa maiores que 9 e menores
que 16 são representados pelas letras A,B,...,F. Logo, obtemos como
resultado:
--------------- // ---------------
2.2.2.1.Código ASCII
2.2.2.2.Método BCD
2.2.2.3.Representação de ponto flutuante
Página 7
assembly
2.2.2.1.Código ASCII
Deste modo, cada letra, dígito ou caracter especial ocupa 1 byte na memória
do computador.
2.2.2.2.Método BCD
Neste método grupos de 4 bits são usados para representar cada dígito
decimal de 0 a 9. Com este método podemos representar 2 dígitos por byte de
informação.
Vemos que este método vem a ser muito mais prático para representação
numérica do que o código ASCII. Embora ainda menos prático do que o binário,
com o método BCD podemos representar dígitos de 0 a 99. Com o binário, vemos
que o alcance é maior, de 0 a 255.
O expoente negativo, por outro lado, indica o número de casas que o ponto
decimal deve se locomover para a esquerda.
Página 8
assembly
--------------- // ---------------
2.3.PROGRAMA DEBUG
Conteúdo:
2.3.2.Registradores da CPU.
AX Registrador Acumulador
BX Registrador Base
CX Registrador Contador
DX Registrador de Dados
DS Registrador de Segmento de Dados
ES Registrador de Segmento Extra
Página 9
assembly
SS Registrador de Segmento de Pilha
CS Registrador de Segmento de Código
BP Registrador Apontador da Base
SI Registrador de Índice Fonte
DI Registrador de Índice Destino
SP Registrador Apontador de Pilha
IP Registrador Apontador da Próxima Instrução
F Registrador de Flag
2.3.3.Programa Debug.
Debug pode apenas criar arquivos com a extensão .COM, e por causa das
características deste tipo de programa, eles não podem exceder os 64 Kb, e
também devem iniciar no endereço de memória 0100H dentro do segmento
específico. É importante observar isso, pois deste modo os programas .COM
não são relocáveis.
C:/>Debug [Enter]
-
-r[Enter]
-rbx
BX 0000
:
2.3.4.Estrutura Assembly.
add ah bh
Aqui "add" é o comando a ser executado, neste caso uma adição, e "ah" bem
como "bh" são os parâmetros.
Por exemplo:
mov al, 25
add al,[170]
Página 11
assembly
Embora neste momento não seja necessário darmos um parâmetro ao comando "a",
isso é recomendável para evitar problemas, logo:
a 100[enter]
mov ax,0002[enter]
mov bx,0004[enter]
add ax,bx[enter]
nop[enter][enter]
O que o programa faz? Move o valor 0002 para o registrador ax, move o valor
0004 para o registrador bx, adiciona o conteúdo dos registradores ax e bx,
guardando o resultado em ax e finalmente a instrução nop (nenhuma operação)
finaliza o programa.
C:\>debug
-a 100
0D62:0100 mov ax,0002
0D62:0103 mov bx,0004
0D62:0106 add ax,bx
0D62:0108 nop
0D62:0109
-t
Vemos o valor 0002 no registrador AX. Teclamos "t" para executar a segunda
instrução:
Página 12
assembly
-t
-t
Não seria prático ter que digitar o programa cada vez que iniciássemos o
Debug. Ao invés disso, podemos armazená-lo no disco. Só que o mais
interessante nisso é que um simples comando de salvar cria um arquivo com a
extensão .COM, ou seja, executável - sem precisarmos efetuar os processos de
montagem e ligação, como veremos posteriormente com o TASM.
Para obter o tamanho de um programa, o comando "h" é usado, já que ele nos
mostra a adição e subtração de dois números em hexadecimal. Para obter o
tamanho do programa em questão, damos como parâmetro o valor do endereço
final do nosso programa (10A), e o endereço inicial (100). O primeiro
resultado mostra-nos a soma dos endereços, o segundo, a subtração.
Página 13
assembly
-h 10a 100
020a 000a
-n test.com
-rcx
CX 0000
:000a
-w
Writing 000A bytes
-n test.com
-l
-u 100 109
0C3D:0100 B80200 MOV AX,0002
0C3D:0103 BB0400 MOV BX,0004
0C3D:0106 01D8 ADD AX,BX
0C3D:0108 CD20 INT 20
O último comando "u" é usado para verificar que o programa foi carregado na
memória. O que ele faz é desmontar o código e mostrá-lo em assembly. Os
parâmetros indicam ao Debug os endereços inicial e final a serem
desmontados.
*****************************************************************************
Página 14
assembly
Conteúdo:
--------------- // ---------------
3.1.1.Software necessário
3.1.2.Programação Assembly
3.1.1.SOFTWARE NECESSÁRIO
O editor pode ser qualquer um que dispusermos. O montador será o TASM macro
assembler da Borland, e o linker será o TLINK, também da Borland.
Nós devemos criar os programas fonte com a extensão .ASM para que o TASM
reconheça e o transforme no programa objeto, um "formato intermediário" do
programa, assim chamado porque ainda não é um programa executável e tão
pouco um programa fonte. O linker gera a partir de um programa .OBJ, ou da
combinação de vários deles, um programa executável, cuja extensão é
normalmente .EXE, embora possa ser .COM dependendo da forma como for montado
e ligado.
3.1.2.PROGRAMAÇÃO ASSEMBLY
.MODEL SMALL
Define o melo de memória a usar em nosso programa
.CODE
Define as instruções do programa, relacionado ao segmento de código
.STACK
Reserva espaço de memória para as instruções de programa na pilha
Página 15
assembly
END
Finaliza um programa assembly
Vamos programar
Primeiro passo
Use qualquer editor para criar o programa fonte. Entre com as seguintes
linhas:
Primeiro exemplo
Segundo passo
Terceiro passo
Exemplo:
C:\>tasm exam1.asm
Turbo Assembler Version 2.0 Copyright (c) 1988, 1990 Borland International
O TASM só pode criar programas no formato .OBJ, que ainda não pode ser
executado...
Quarto passo
Página 16
assembly
Exemplo:
C:\>tlink exam1.obj
Turbo Link Version 3.0 Copyright (c) 1987, 1990 Borland International
C:\>
Quinto passo
C:\>exam1[enter]
--------------- // ---------------
3.2.Processo Assembly.
3.2.1.Segmentos
3.2.2.Tabela de equivalência
3.2.1.SEGMENTOS
Desta forma, para que o montador seja capaz de gerenciar os dados, se faz
necessário que cada informação ou instrução se encontre na área
correspondente ao seu segmento. O endereço do segmento é fornecido ao
montador pelos registradores DS, ES, SS e CS. Lembrando um programa no
Debug, observe:
O modo usado para indicar ao montador com quais segmentos vamos trabalhar é
fazendo uso das diretivas .CODE, .DATA e .STACK.
3.2.2.TABELAS DE EQUIVALÊNCIA
Cada uma das partes numa linha de código assembly é conhecida como token,
por exemplo:
MOV AX,Var
Há símbolos que o montador não consegue encontrar, uma vez que podem ser
declarações externas. Neste caso o linker entra em ação para criar a
estrutura necessária a fim de ligar as diversas possíveis partes de código,
dizendo ao loader que o segmento e o token em questão são definidos quando o
programa é carregado e antes de ser executado.
--------------- // ---------------
3.3.Mais programas.
Página 18
assembly
Outro exemplo
Primeiro passo
;exemplo2
.model small
.stack
.code
mov ah,2h ;move o valor 2h para o registrador ah
mov dl,2ah ;move o valor 2ah para o registrador dl
;(é o valor ASCII do caractere *)
int 21h ;interrupção 21h
mov ah,4ch ;função 4ch, sai para o sistema operacional
int 21h ;interrupção 21h
end ;finaliza o programa
Segundo passo
Terceiro passo
C:\>tasm exam2.asm
Turbo Assembler Version 2.0 Copyright (c) 1988, 1990 Borland International
Quarto passo
C:\>tlink exam2.obj
Turbo Link Version 3.0 Copyright (c) 1987, 1990 Borland International
C:\>
Quinto passo
Executar o programa:
C:\>exam2[enter]
*
Página 19
assembly
C:\>
--------------- // ---------------
3.4.Tipos de instruções.
3.4.1.Movimento de dados
3.4.2.Operações lógicas e aritméticas
3.4.3.Saltos, laços e procedimentos
3.4.1.MOVIMENTO DE DADOS
É possível mover blocos de dados através de instruções movs, que copia uma
cadeia de bytes ou palavras; movsb copia n bytes de um local para outro; e
movsw copia n palavras. A última das duas instruções toma os valores dos
endereços definidos por DS:SI como o grupo de dados a mover e ES:DI como a
nova localização dos dados.
PUSH AX
PUSH BX
PUSH CX
POP CX
Página 20
assembly
POP BX
POP AX
OUT DX,AX
IN AX,DX
As instruções usadas para operações algébricas são: para adição add, para
subtração sub, para multiplicação mul e para divisão div.
Página 21
assembly
*****************************************************************************
Conteúdo:
--------------- // ---------------
Conteúdo:
4.1.1.Instruções de transferência
4.1.2.Instruções de carga
4.1.3.Instruções de pilha
4.1.1.Instruções de transferência.
São usadas para mover o conteúdo dos operadores. Cada instrução pode ser
usada com diferentes modos de endereçamento.
MOV
MOVS (MOVSB) (MOVSW)
INSTRUÇÃO MOV
Sintaxe:
MOV Destino,Fonte
Destino é o lugar para onde o dado será movido e Fonte é o lugar onde o dado
está.
Exemplo:
MOV AX,0006h
MOV BX,AX
MOV AX,4C00h
INT 21h
Este pequeno programa move o valor 0006h para o registrador AX, então ele
move o conteúdo de AX (0006h) para o registrador BX, e finalmente move o
valor 4C00h para o registrador AX para terminar a execução com a opção 4C da
interrupção 21h.
Sintaxe:
MOVS
Este comando não necessita de parâmetros uma vez que toma como endereço
fonte o conteúdo do registrador SI e como destino o conteúdo de DI. A
seguinte seqüência de instruções ilustra isso:
As instruções MOVSB e MOVSW são usadas do mesmo modo que MOVS, a primeira
move um byte e a segunda move uma palavra.
Instruções de carga.
Página 23
assembly
Sintaxe:
LODS
INSTRUÇÃO LAHF
Sintaxe:
LAHF
Esta instrução é útil para verificar o estado dos flags durante a execução
do nosso programa.
SF ZF ?? AF ?? PF ?? CF
INSTRUÇÃO LDS
Sintaxe:
LDS destino,fonte
Página 24
assembly
O operador fonte deve ser uma double word na memória. A palavra associada
com o maior endereço é transferida para DS, em outras palavras isto é tomado
como o endereço de segmento. A palavra associada com o menor endereço é o
endereço de deslocamento e é depositada no registrador indicado como
destino.
INSTRUÇÃO LEA
Sintaxe:
LEA destino,fonte
Para ilustrar uma das facilidades que temos com este comando, vejamos:
É equivalente a:
LEA SI,VAR1
É muito provável que para o programador é muito mais fácil criar programas
grandes usando este último formato.
INSTRUÇÃO LES
Sintaxe:
LES destino,fonte
POP
Página 25
assembly
POPF
PUSH
PUSHF
INSTRUÇÃO POP
Sintaxe:
POP destino
Este incremento é duplo pelo fato de que a pilha do mais alto endereço de
memória para o mais baixo, e a pilha trabalha apenas com palavras, 2 bytes,
logo deve ser 2 o incremento de SP, na realidade 2 está sendo subtraído do
tamanho real da pilha.
INSTRUÇÃO POPF
Sintaxe:
POPF
BIT FLAG
0 CF
2 PF
4 AF
6 ZF
7 SF
8 TF
9 IF
10 DF
11 OF
INSTRUÇÃO PUSH
Sintaxe:
PUSH fonte
INSTRUÇÃO PUSHF
Sintaxe:
PUSHF
--------------- // ---------------
Conteúdo:
4.2.1.Instruções lógicas
4.2.2.Instruções aritméticas
4.2.1.Instruções lógicas
AND
NEG
NOT
OR
TEST
XOR
INSTRUÇÃO AND
Sintaxe:
AND destino,fonte
Com esta instrução a operação lógica "y" para ambos os operadores é usada
como na tabela:
INSTRUÇÃO NEG
Sintaxe:
NEG destino
NEG AX
Isto fará com o que o valor EDCCH fque armazenado no registrador AX.
INSTRUÇÃO NOT
Sintaxe:
Página 28
assembly
NOT destino
INSTRUÇÃO OR
Sintaxe:
OR destino,fonte
A instrução OR, faz uma disjunção lógica bit a bit dos dois operadores:
INSTRUÇÃO TEST
Sintaxe:
TEST destino,fonte
Realiza uma conjunção, bit a bit, dos operadores, mas difere da instrução
AND, uma vez que não coloca o resultado no operador de destino. Tem efeito
sobre o registrador de flag.
INSTRUÇÃO XOR
Sintaxe:
XOR destino,fonte
4.2.2.Instruções aritméticas.
ADC
ADD
DIV
IDIV
MUL
IMUL
SBB
SUB
INSTRUÇÃO ADC
Sintaxe:
ADC destino,fonte
Esta instrução efetua a soma entre dois operandos, mais o valor do flag CF,
existente antes da operação. Apenas o operando destino e os flags são
afetados.
INSTRUÇÃO ADD
Sintaxe:
ADD destino,fonte
Página 30
assembly
INSTRUÇÃO DIV
Sintaxe:
DIV fonte
INSTRUÇÃO IDIV
Sintaxe:
IDIV fonte
INSTRUÇÃO MUL
Sintaxe:
MUL fonte
INSTRUÇÃO IMUL
Página 31
assembly
Sintaxe:
IMUL fonte
Esta instrução faz o mesmo que a anterior, difere apenas pela inclusão do
sinal.
INSTRUÇÃO SBB
Sintaxe:
SBB destino,fonte
INSTRUÇÃO SUB
Propósito: Subtração.
Sintaxe:
SUB destino,fonte
--------------- // ---------------
Conteúdo:
4.3.1.Instruções de salto
4.3.2.Instruções de laços: loop
4.3.3.Instruções de contagem
Página 32
assembly
4.3.4.Instruções de comparação
4.3.5.Instruções de flag
4.3.1.Instruções de salto.
JMP
JA (JNBE)
JAE (JNBE)
JB (JNAE)
JBE (JNA)
JE (JZ)
JNE (JNZ)
JG (JNLE)
JGE (JNL)
JL (JNGE)
JLE (JNG)
JC
JNC
JNO
JNP (JPO)
JNS
JO
JP (JPE)
JS
INSTRUÇÃO JMP
Sintaxe:
JMP destino
Esta instrução é usada par adesviar o curso do programa sem tomar em conta
as condições atuais dos flags ou dos dados.
INSTRUÇÃO JA (JNBE)
Sintaxe:
JA símbolo
Página 33
assembly
Após uma comparação este comando salta se não é igual.
Sintaxe:
JAE símbolo
INSTRUÇÃO JB (JNAE)
Sintaxe:
JB símbolo
Sintaxe:
JBE símbolo
INSTRUÇÃO JE (JZ)
Sintaxe:
JE símbolo
Sintaxe:
JNE símbolo
INSTRUÇÃO JG (JNLE)
Sintaxe:
JG símbolo
Sintaxe:
JGE símbolo
Página 35
assembly
INSTRUÇÃO JL (JNGE)
Sintaxe:
JL símbolo
A instrução salta se está less than, se está not larger than ou se está
equal.
Sintaxe:
JLE símbolo
A instrução salta se está less than, se está equal ou se está not larger.
INSTRUÇÃO JC
Sintaxe:
JC símbolo
O salto é feito se CF = 1.
INSTRUÇÃO JNC
Sintaxe:
JNC símbolo
Página 36
assembly
A instrução salta se não há carry.
O salto é feito se CF = 0.
INSTRUÇÃO JNO
Sintaxe:
JNO símbolo
O salto é feito se OF = 0.
Sintaxe:
JNP símbolo
O salto é feito se PF = 0.
INSTRUÇÃO JNS
Sintaxe:
JNP símbolo
O salto é feito se SF = 0.
INSTRUÇÃO JO
Página 37
assembly
Sintaxe:
JO símbolo
O salto é feito se OF = 1.
INSTRUÇÃO JP (JPE)
Sintaxe:
JP símbolo
O salto é feito se PF = 1.
INSTRUÇÃO JS
Sintaxe:
JS símbolo
--------------- // ---------------
LOOP
LOOPE
LOOPNE
Página 38
assembly
INSTRUÇÃO LOOP
Sintaxe:
LOOP símbolo
INSTRUÇÃO LOOPE
Sintaxe:
LOOPE símbolo
INSTRUÇÃO LOOPNE
Sintaxe:
LOOPNE símbolo
--------------- // ---------------
4.3.3.Instruções contadoras.
DEC
INC
Página 39
assembly
DEC INSTRUCTION
Sintaxe:
DEC destino
INSTRUÇÃO INC
Sintaxe:
INC destino
--------------- // ---------------
4.3.4.Instruções de comparação.
CMP
CMPS (CMPSB) (CMPSW)
INSTRUÇÃO CMP
Sintaxe:
CMP destino,fonte
Página 40
assembly
Propósito: Comparar cadeias de um byte ou uma palavra.
Sintaxe:
CMP destino,fonte
--------------- // ---------------
4.3.5.Instruções de flag.
CLC
CLD
CLI
CMC
STC
STD
STI
INSTRUÇÃO CLC
Sintaxe:
CLC
Página 41
assembly
INSTRUÇÃO CLD
Sintaxe:
CLD
INSTRUÇÃO CLI
Sintaxe:
CLI
INSTRUÇÃO CMC
Sintaxe:
CMC
INSTRUÇÃO STC
Sintaxe:
STC
INSTRUÇÃO STD
Sintaxe:
STD
INSTRUÇÃO STI
Sintaxe:
STI
*****************************************************************************
Conteúdo:
5.1.Interrupções
5.2.Gerenciamento de arquivos
--------------- // ---------------
Conteúdo
Interrupções internas são geradas por certos eventos que ocorrem durante a
execução de um programa.
Página 43
assembly
Não nos é permitido gerenciar diretamente esta interrupção, uma vez que não
se pode controlar a hora atualizada por software. Mas podemos usar seus
efeitos no computador para o nosso benefício, por exemplo para criar um
virtual clock atualizado continuamente pelo contador interno de clock. Para
tanto, precisamos apenas ler o valor atual do contador e o transformar num
formato compreensível pelo usuário.
--------------- // ---------------
Estas interrupções não são enviadas diretamente para a CPU, mas, de uma
forma melhor, são enviadas para um circuito integrado cuja função exclusiva
é manusear este tipo de interrupção. O circuito, chamado PIC8259A, é
controlado pela CPU através de uma série de comunicação chamada paths.
--------------- // ---------------
5.1.3.Interrupções de software
Página 44
assembly
A escolha sobre qual o tipo de interrupção usar irá depender somente das
características que você deseja dar ao seu programa: velocidade (use BIOS),
portabilidade (use DOS).
--------------- // ---------------
Conteúdo
--------------- // ---------------
5.1.4.1.Interrupção 21H
Sintaxe:
Int 21H
Esta interrupção tem muitas funções, para acessar cada uma delas é
necessário que o número correspondente da função esteja no registrador AH no
momento da chamada da interrupção.
Nesta seção são apenas especificadas as tarefas de cada função, para uma
referência acerca dos conceitos usados, veja Introdução ao gerenciamento de
arquivos.
Método FCB
Handles
FUNÇÃO 02H
Uso:
Registradores de chamada:
AH = 02H
DL = Valor de caracter a ser mostrado.
Registradores de retorno:
Nenhum.
FUNÇÃO 09H
Página 46
assembly
Uso:
Registradores de chamada:
AH = 09H
DS:DX = Endereço de início da cadeia de caracteres.
Registradores de retorno:
Nenhum.
FUNÇÃO 40H
Uso:
Registradores de chamada:
AH = 40H
BX = Número do handle
CX = Quantidade de bytes a gravar
DS:DX = Área onde está o dado
Registradores de retorno:
CF = 1 se houve erro
AX = Código de erro
Para usar esta função para mostrar a informação na tela, faça o registrador
BX ser igual a 1, que é o valor default para o vídeo no DOS.
FUNÇÃO 01H
Página 47
assembly
Uso:
Registradores de chamada
AH = 01H
Registradores de retorno:
AL = Caracter lido
FUNÇÃO 0AH
Uso:
Registradores de chamada:
AH = 0AH
DS:DX = Endereço inicial da área de armazenamento
BYTE 0 = Quantidade de bytes na área
BYTE 1 = Quantidade de bytes lidos
do BYTE 2 até BYTE 0 + 2 = caracteres lidos
Registradores de retorno:
Nenhum.
FUNÇÃO 3FH
Uso:
Página 48
assembly
Registradores de chamada:
AH = 3FH
BX = Número do handle
CX = Número de bytes a ler
DS:DX = Área para receber o dado
Registradores de retorno:
FUNÇÃO 0FH
Uso:
Registradores de chamada:
AH = 0FH
DS:DX = Ponteiro para um FCB
Registradores de retorno:
FUNÇÃO 14H
Uso:
Registradores de chamada:
AH = 14H
DS:DX = Ponteiro para um FCB já aberto.
Registradores de retorno:
O que esta função faz é ler o próximo bloco de informações do endereço dado
Página 49
assembly
por DS:DX, e atualizar este registro.
FUNÇÃO 15H
Uso:
Registradores de chamada:
AH = 15H
DS:DX = Ponteiro para um FCB ja aberto.
Registradores de retorno:
FUNÇÃO 16H
Uso:
AH = 16H
DS:DX = Ponteiro para um FCB já aberto.
Registradores de retorno:
FUNÇÃO 21H
Uso:
Registradores de chamada:
AH = 21H
Página 50
assembly
DS:DX = Ponteiro para FCB aberto.
Registradores de retorno:
FUNÇÃO 22H
Uso:
Registradores de chamada:
AH = 22H
DS:DX = Ponteiro para um FCB aberto.
Registradores de retorno:
FUNÇÃO 3CH
Uso:
Registradores de chamada:
AH = 3CH
CH = Atributo do arquivo
DS:DX = Nome do arquivo, no formato ASCII.
Registradores de retorno:
Valor Atributos
00H Normal
02H Hidden
04H System
06H Hidden e System
FUNÇÃO 3DH
Uso:
Registradores de chamada:
AH = 3DH
AL = modo de acesso
DS:DX = Nome do arquivo, no formato ASCII.
Registradores de retorno:
BITS
7 6 5 4 3 2 1
. . . . 0 0 0 Apenas leitura
. . . . 0 0 1 Apenas escrita
Página 52
assembly
. . . . 0 1 0 Leitura/Escrita
. . . x . . . RESERVADO
FUNÇÃO 3EH
Uso:
Registradores de chamada:
AH = 3EH
BX = Número do handle associado
Registradores de retorno:
FUNÇÃO 3FH
Uso:
--------------- // ---------------
5.1.4.2.Interrupção 10h
Sintaxe:
Int 10H
Esta interrupção tem várias funções, todas para entrada e saída de vídeo.
Para acessar cada uma delas é necessário colocar o número da função
correspondente no registrador AH.
Página 53
assembly
Função 02h
Uso:
Registradores de chamada:
AH = 02H
DH = linha
DL = coluna
Registradores de retorno:
Nenhum.
Função 09h
Uso:
Registradores de chamada:
AH = 09H
AL = Caracter a exibir
BH = Página de vídeo, onde o caracter será mostrado
BL = Atributo do caracter
CX = Número de repetições.
Registradores de retorno:
Nenhum
Esta função mostra um caracter na tela várias vezes, de acordo com o número
especificado no registrador CX, mas sem mudar a posição do cursor na tela.
Página 54
assembly
Função 0Ah
Uso:
Registradores de chamada:
AH = 0AH
AL = Caracter a exibir
BH = Página de vídeo onde o caracter será exibido
BL = Cor do caracter (apenas em modo gráfico)
CX = Número de repetições
Registradores de retorno:
Nenhum.
Função 0EH
Uso:
Registradores de chamada:
AH = 0EH
AL = Caracter a exibir
BH = Página de vídeo onde o caracter será exibido
BL = Cor a usar (apenas em modo gráfico)
Registradores de retorno:
Nenhum
--------------- // ---------------
5.1.4.3.Interrupção 16H
Página 55
assembly
Função 00H, lê um caracter do teclado.
Função 01H, lê o estado atual do teclado.
Registradores de chamada:
AH = 00H
Registradores de retorno:
Quando se usa esta interrupção, os programas executam até que uma tecla seja
pressionada. Se é um valor ASCII, é armazenado no registrador AH. Caso
contrário, o código é armazenado no registrador AL e AH=0.
Este valor de AL pode ser utilizado quando queremos detectar teclas que não
estão diretamente representadas pelo seu valor ASCII, tais como
[ALT][CONTROL].
Função 01h
Uso:
Registradores de chamada:
AH = 01H
Registradores de retorno:
--------------- // ---------------
5.1.4.4.Interrupção 17H
Sintaxe:
Página 56
assembly
Int 17H
Função 00H
Uso:
Registradores de chamada:
AH = 00H
AL = Caracter a imprimir
DX = Porta de conexão
Registradores de retorno:
AH = Estado da impressora
A maioria dos BIOS suportam 3 portas paralelas, havendo alguns que suportam
4.
Função 01h
Uso:
Página 57
assembly
Registradores de chamada:
AH = 01H
DX = Porta
Registradores de retorno:
AH = Status da impressora
Função 02h
Uso:
Registradores de chamada:
AH = 01H
DX = Porta
Registradores de retorno
AH = Status da impressora
--------------- // ---------------
Conteúdo:
Para uma melhor facilidade, daqui por diante nos referiremos aos Blocos de
Controle de Arquivo como FCBs e aos Canais de Comunicação como handles.
--------------- // ---------------
5.2.2.Método FCB.
5.2.2.1.Introdução
5.2.2.2.Abertura de arquivo
5.2.2.3.Criar um novo arquivo
5.2.2.4.Escrita seqüencial
Página 59
assembly
5.2.2.5.Leitura seqüencial
5.2.2.6.Leitura e escrita randômica
5.2.2.7.Fechar um arquivo
--------------- // ---------------
5.2.2.1.INTRODUÇÃO
--------------- // ---------------
5.2.2.2.ABERTURA DE ARQUIVO
Página 60
assembly
Para abrir um arquivo FCB é usada a função 0FH da interrupção 21h.
A unidade, o nome e a extensão do arquivo devem ser inicializadas antes da
abertura.
--------------- // ---------------
Caso ocorra problema, o valor FFH deve retornar em AL, de outra forma este
registrador conterá o valor 0.
--------------- // ---------------
5.2.2.4.ESCRITA SEQÜENCIAL
A função 1AH não retorna qualquer estado do disco nem da operação. Mas a
função 15H, que usaremos para escrever para o disco, faz isso no registrador
AL. Se este for igual a zero, então não há erro e os campos de registro
corrente e de bloco são atualizados.
--------------- // ---------------
5.2.2.5.LEITURA SEQÜENCIAL
--------------- // ---------------
--------------- // ---------------
5.2.2.7.FECHAR UM ARQUIVO
--------------- // ---------------
5.2.3.Canais de comunicação.
Quando usamos este método para trabalhar com arquivos, não há distinção
entre acesso seqüencial ou randômico, o arquivo é simplesmente tomado como
uma rede de bytes.
--------------- // ---------------
*****************************************************************************
Conteúdo
6.1.Procedimentos
6.2.Macros
6.1.Procedimentos
6.1.1.Definição de procedimento
6.1.2.Sintaxe de um procedimento
--------------- // ---------------
6.1.1.Definição de um procedimento
--------------- // ---------------
6.1.2.Sintaxe de um procedimento
CALL NomedoProcedimento
Página 63
assembly
As partes que compõem um procedimento são as seguintes:
Declaração do procedimento
Código do procedimento
Diretiva de retorno
Término do procedimento
Por exemplo, se quisermos uma rotina que soma dois bytes armazenados em AH e
AL, e o resultado da soma em BX:
Call Soma
--------------- // ---------------
6.2.Macros
--------------- // ---------------
Declaração da macro
Código da macro
Diretiva de término da macro
Para usar uma macro basta chamá-la pelo seu nome, tal como se fosse qualquer
instrução na linguagem assembly:
Pos 8, 6
Página 65
assembly
--------------- // ---------------
6.2.3.Biblioteca de Macros
Para incluir uma biblioteca num programa, basta colocar a seguinte instrução
Include NomedoArquivo na parte inicial do programa, antes da declaração do
modelo de memória.
Supondo que o arquivo de macros tenha sido salvo com o nome de MACROS.TXT, a
instrução Include seria utilizada do seguinte modo:
;Início do programa
Include MACROS.TXT
.MODEL SMALL
.DATA
;Os dados vão aqui
.CODE
Inicio:
;O código do programa começa aqui
.STACK
;A pilha é declarada
End Inicio
;Fim do programa
*****************************************************************************
Conteúdo:
--------------- // ---------------
Procedimento
Página 66
assembly
Primeiro passo
Por exemplo:
C:\>debug
-n one.com
-l
-u 100 109
0D80:0100 B80600 MOV AX,0006
0D80:0103 BB0400 MOV BX,0004
0D80:0106 01D8 ADD AX,BX
0D80:0108 CD20 INT 20
-
Nota:
-n one.com
-l
Carregá-lo
-u 100 109
Segundo passo
Digite o comando g
Por exemplo:
-g
Primeiro exemplo
-a0100
297D:0100 MOV AX,0006 ;Põe o valor 0006 no registrador AX
297D:0103 MOV BX,0004 ;Põe o valor 0004 no registrador BX
297D:0106 ADD AX,BX ;Adiciona BX ao conteúdo de AX
297D:0108 INT 20 ;Finaliza o Programa
Página 67
assembly
A única coisa que este programa faz é salvar dois valores em dois
registradores e adicionar o valor de um ao outro.
Segundo exemplo
- a100
0C1B:0100 jmp 125 ;Salta para o endereço 125h
0C1B:0102 [Enter]
- e 102 'Hello, How are you ?' 0d 0a '$'
- a125
0C1B:0125 MOV DX,0102 ;Copia a string para registrador DX
0C1B:0128 MOV CX,000F ;Quantas vezes a string será mostrada
0C1B:012B MOV AH,09 ;Copia o valor 09 para registrador AH
0C1B:012D INT 21 ;Mostra a string
0C1B:012F DEC CX ;Subtrai 1 de CX
0C1B:0130 JCXZ 0134 ;Se CX é igual a 0 salta para o endereço 0134
0C1B:0132 JMP 012D ;Salta ao endereço 012D
0C1B:0134 INT 20 ;Finaliza o programa
Terceiro exemplo
-a100
297D:0100 MOV AH,01 ;Função para mudar o cursor
297D:0102 MOV CX,0007 ;Formata o cursor
297D:0105 INT 10 ;Chama interrupção do BIOS
297D:0107 INT 20 ;Finaliza o programa
Quarto exemplo
-a100
297D:0100 MOV AH,01 ;Função 1 (lê caractere do teclado)
297D:0102 INT 21 ;Chama interrupção do DOS
297D:0104 CMP AL,0D ;Compara se o caractere lido é um ENTER
297D:0106 JNZ 0100 ;Se não é, lê um outro caractere
297D:0108 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:010A MOV DL,AL ;Character to write on AL
297D:010C INT 21 ;Chama interrupção do DOS
297D:010E INT 20 ;Finaliza o programa
Este programa usa a interrupção 21h do DOS. Usa duas funções da mesma: a
primeira lê um caractere do teclado (função 1) e a segundo escreve um
caractere na tela. O programa lê caracteres do teclado até encontrar um
ENTER.
Quinto exemplo
-a100
297D:0100 MOV AH,02 ;Função 2 (escreve um caractere na tela)
Página 68
assembly
297D:0102 MOV CX,0008 ;Põe o valor 0008 no registrador CX
297D:0105 MOV DL,00 ;Põe o valor 00 no registrador DL
297D:0107 RCL BL,1 ;Rotaciona o byte em BL um bit para a esquerda
297D:0109 ADC DL,30 ;Converte o registrador de flag para 1
297D:010C INT 21 ;Chama interrupção do DOS
297D:010E LOOP 0105 ;Salta se CX > 0 para o endereço 0105
297D:0110 INT 20 ;Finaliza o programa
Sexto exemplo
-a100
297D:0100 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:0102 MOV DL,BL ;Põe o valor de BL em DL
297D:0104 ADD DL,30 ;Adiciona o valor 30 a DL
297D:0107 CMP DL,3A ;Compara o valor 3A com o conteúdo de DL sem
afetá-lo
;seu valor apenas modifica o estado do flag de
carry
297D:010A JL 010F ;salta ao endereço 010f, se for menor
297D:010C ADD DL,07 ;Adiciona o valor 07 a DL
297D:010F INT 21 ;Chama interrupção do DOS
297D:0111 INT 20 ;Finaliza o programa
Sétimo exemplo
-a100
297D:0100 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:0102 MOV DL,BL ;Põe o valor de BL em DL
297D:0104 AND DL,0F ;Transporta fazendo AND dos números bit a bit
297D:0107 ADD DL,30 ;Adiciona 30 a Dl
297D:010A CMP DL,3A ;Compara Dl com 3A
297D:010D JL 0112 ;Salta ao endereço 0112, se menor
297D:010F ADD DL,07 ;Adiciona 07 a DL
297D:0112 INT 21 ;Chama interrupção do DOS
297D:0114 INT 20 ;Finaliza o programa
Oitavo exemplo
-a100
297D:0100 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:0102 MOV DL,BL ;Põe o valor de BL em DL
297D:0104 MOV CL,04 ;Põe o valor 04 em CL
297D:0106 SHR DL,CL ;Desloca os 4 bits mais altos do número ao nibble
mais à direita
297D:0108 ADD DL,30 ;Adiciona 30 a DL
Página 69
assembly
297D:010B CMP DL,3A ;Compara Dl com 3A
297D:010E JL 0113 ;Salta ao endereço 0113, se menor
297D:0110 ADD DL,07 ;Adiciona 07 a DL
297D:0113 INT 21 ;Chama interrupção do DOS
297D:0115 INT 20 ;Finaliza o programa
Nono exemplo
-a100
297D:0100 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:0102 MOV DL,BL ;Põe o valor de BL em DL
297D:0104 MOV CL,04 ;Põe o valor 04 em CL
297D:0106 SHR DL,CL ;Desloca os 4 bits mais altos do número ao nibble
mais à direita
297D:0108 ADD DL,30 ;Adiciona 30 a DL
297D:010B CMP DL,3A ;Compara Dl com 3A
297D:010E JL 0113 ;Salta ao endereço 0113, se menor
297D:0110 ADD DL,07 ;Adiciona 07 a DL
297D:0113 INT 21 ;Chama interrupção do DOS
297D:0115 MOV DL,BL ;Põe o valor de BL em DL
297D:0117 AND DL,0F ;Transporta fazendo AND dos números bit a bit
297D:011A ADD DL,30 ;Adiciona 30 a DL
Décimo exemplo
-a100
297D:0100 MOV AH,01 ;Função 1 (lê caractere do teclado)
297D:0102 INT 21 ;Chama interrupção do DOS
297D:0104 MOV DL,AL ;Põe o valor de AL em DL
297D:0106 SUB DL,30 ;Subtrai 30 de DL
297D:0109 CMP DL,09 ;Compara DL com 09
297D:010C JLE 0111 ;Salta ao endereço 0111, se menor ou igual
297D:010E SUB DL,07 ;Subtrai 07 de DL
297D:0111 MOV CL,04 ;Põe o valor 04 em CL
297D:0113 SHL DL,CL ;Insere zeros à direita
297D:0115 INT 21 ;Chama interrupção do DOS
297D:0117 SUB AL,30 ;Subtrai 30 de AL
297D:0119 CMP AL,09 ;Compara AL com 09
297D:011B JLE 011F ;Salta ao endereço 011f, se menor ou igual
297D:011D SUB AL,07 ;Subtrai 07 de AL
297D:011F ADD DL,AL ;Adiciona AL a DL
297D:0121 INT 20 ;Finaliza o programa
Página 70
assembly
-a100
297D:0100 CALL 0200 ;Chama um procedimento
297D:0103 INT 20 ;Finaliza o programa
-a200
297D:0200 PUSH DX ;Põe o valor de DX na pilha
297D:0201 MOV AH,08 ;Função 8
297D:0203 INT 21 ;Chama interrupção do DOS
297D:0205 CMP AL,30 ;Compara AL com 30
297D:0207 JB 0203 ;Salta se CF é ativado ao endereço 0203
297D:0209 CMP AL,46 ;Compara AL com 46
297D:020B JA 0203 ;Salta ao endereço 0203, se diferente
297D:020D CMP AL,39 ;Compara AL com 39
297D:020F JA 021B ;Salta ao endereço 021B, se diferente
297D:0211 MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:0213 MOV DL,AL ;Põe o valor de AL em DL
297D:0215 INT 21 ;Chama interrupção do DOS
297D:0217 SUB AL,30 ;Subtrai 30 de AL
297D:0219 POP DX ;Extrai o valor de DX da pilha
297D:021A RET ;Retorna o controle ao programa principal
297D:021B CMP AL,41 ;Compara AL com 41
297D:021D JB 0203 ;Salta se CF é ativado ao endereço 0203
297D:021F MOV AH,02 ;Função 2 (escreve um caractere na tela)
297D:022 MOV DL,AL ;Põe o valor AL em DL
297D:0223 INT 21 ;Chama interrupção do DOS
297D:0225 SUB AL,37 ;Subtrai 37 de AL
297D:0227 POP DX ;Extrai o valor de DX da pilha
297D:0228 RET ;Retorna o controle ao programa principal
Este programa se mantém lendo caracteres até receber um que possa ser
convertido para um número hexadecimal.
--------------- // ---------------
Procedimento:
Primeiro passo
Montar o programa
Página 71
assembly
Por exemplo:
C:\>tasm one.asm
Turbo Assembler Version 2.0 Copyright (c) 1988, 1990 Borland International
C:\>
Isto criará um programa objeto com o mesmo nome do fonte, neste caso:
one.obj
Segundo passo
C:\>tlink one.obj
Turbo Link Version 3.0 Copyright (c) 1987, 1990 Borland International
C:\>
Isto cria o programa executável com o mesmo nome do objeto e com extensão
diferente, one.exe
Terceiro passo
Primeiro exemplo
Segundo exemplo
WRITE_CHAR PROC
MOV AH,2h ;Função 2h, imprime caracter
INT 21h ;Imprime o caracter que está em DL
RET ;Retorna o controle ao procedimento que chamou
WRITE_CHAR ENDP ;Finaliza o procedimento
END PRINT_A_J ;Finaliza o programa
Terceiro exemplo
TEST_WRITE_HEX PROC
MOV DL,3Fh ;Move o valor 3Fh para o registrador DL
CALL WRITE_HEX ;Chama a sub-rotina
Página 73
assembly
MOV AH,4CH ;Função 4Ch
INT 21h ;Retorna o controle ao DOS
TEST_WRITE_HEX ENDP ;Finaliza o procedimento
PUBLIC WRITE_HEX
;........................................................;
;Este procedimento converte para hexadecimal o byte ;
;armazenado no registrador DL e mostra o dígito ;
;Use:WRITE_HEX_DIGIT ;
;........................................................;
WRITE_HEX PROC
PUSH CX ;coloca na pilha o valor do registrador CX
PUSH DX ;coloca na pilha o valor do registrador DX
MOV DH,DL ;move o valor do registrador DL para o registrador DH
MOV CX,4 ;move o valor 4 para o registrador CX
SHR DL,CL
CALL WRITE_HEX_DIGIT ;mostra na tela o primeiro número hexadecimal
MOV DL,DH ;move o valor do registrador DH para o registrador DL
AND DL,0Fh
CALL WRITE_HEX_DIGIT ;mostra na tela o segundo número hexadecimal
POP DX ;retira da pilha o valor do registrador DX
POP CX ;retira da pilha o valor do registrador CX
RET ;Retorna o controle ao procedimento que chamou
WRITE_HEX ENDP
PUBLIC WRITE_HEX_DIGIT
;......................................................................;
;Este procediento converte os 4 bits mais baixos do registrador DL ;
;para um número hexadecimal e o mostrana tela do computador ;
;Use: WRITE_CHAR ;
;......................................................................;
WRITE_HEX_DIGIT PROC
PUSH DX ;coloca na pilha o valor de DX
CMP DL,10 ;compara se o número de bits é menor do que 10
JAE HEX_LETTER ;se não, salta para HEX_LETER
ADD DL,"0" ;se sim, converte para número
JMP Short WRITE_DIGIT ;escreve o caracter
HEX_LETTER:
ADD DL,"A"-10 ;converte um caracter para hexadecimal
WRITE_DIGIT:
CALL WRITE_CHAR ;imprime o caracter na tela
POP DX ;Retorna o valor inicial do registrador DX
;para o registrador DL
RET ;Retorna o controle ao procedimento que chamou
WRITE_HEX_DIGIT ENDP
PUBLIC WRITE_CHAR
;......................................................................;
;Este procedimento imprime um caracter na tela usando o D.O.S. ;
;......................................................................;
Página 74
assembly
WRITE_CHAR PROC
PUSH AX ;Coloca na pilha o valor do registarador AX
MOV AH,2 ;Função 2h
INT 21h ;Interrupção 21h
POP AX ;Extrai da pilha o valor de AX
RET ;Retorna o controle ao procedimento que chamou
WRITE_CHAR ENDP
Quarto exemplo
TEST_WRITE_DECIMAL PROC
MOV DX,12345 ;Move o valor decimal 12345 para o registrador DX
CALL WRITE_DECIMAL ;Chama o procedimento
MOV AH,4CH ;Função 4Ch
INT 21h ;Interrupção 21h
TEST_WRITE_DECIMAL ENDP ;Finaliza o procedimento
PUBLIC WRITE_DECIMAL
;.................................................................;
;Este procedimento escreve um número de 16 bit como um número ;
;sem sinal em notação decimal ;
;Use: WRITE_HEX_DIGIT ;
;.................................................................;
WRITE_DECIMAL PROC
PUSH AX ;Põe na pilha o valor do registrador AX
PUSH CX ;Põe na pilha o valor do registrador CX
PUSH DX ;Põe na pilha o valor do registrador DX
PUSH SI ;Põe na pilha o valor do registrador SI
MOV AX,DX ;move o valor do registrador DX para AX
MOV SI,10 ;move o valor 10 para o registrador SI
XOR CX,CX ;zera o registrador CX
NON_ZERO:
XOR DX,DX ;zera o registrador CX
DIV SI ;divizão entre SI
PUSH DX ;Põe na pilha o valor do registrador DX
INC CX ;incrementa CX
OR AX,AX ;não zero
JNE NON_ZERO ;salta para NON_ZERO
WRITE_DIGIT_LOOP:
POP DX ;Retorna o valor em modo reverso
CALL WRITE_HEX_DIGIT ;Chama o procedimento
LOOP WRITE_DIGIT_LOOP ;loop
END_DECIMAL:
Página 75
assembly
POP SI ;retira da pilha o valor do registrador SI
POP DX ;retira da pilha o valor do registrador DX
POP CX ;retira da pilha o valor do registrador CX
POP AX ;retira da pilha o valor do registrador AX
RET ;Retorna o controle ao procedimento que chamou
WRITE_DECIMAL ENDP ;Finaliza o procedimento
PUBLIC WRITE_HEX_DIGIT
;......................................................................;
; ;
;Este procedimento converte os 4 bits mais baixos do registrador DL ;
;num número hexadecimal e os imprime ;
;Use: WRITE_CHAR ;
;......................................................................;
WRITE_HEX_DIGIT PROC
PUSH DX ;Põe na pilha o valor do registrador DX
CMP DL,10 ;Compara o valor 10 com o valor do registrador DL
JAE HEX_LETTER ;se não, salta para HEX_LETER
ADD DL,"0" ;se é, converte em dígito numérico
JMP Short WRITE_DIGIT ;escreve o caracter
HEX_LETTER:
ADD DL,"A"-10 ;converte um caracter para um número hexadecimal
WRITE_DIGIT:
CALL WRITE_CHAR ;mostra o caracter na tela
POP DX ;Retorna o valor inicial para o registrador DL
RET ;Retorna o controle ao procedimento que chamou
WRITE_HEX_DIGIT ENDP
PUBLIC WRITE_CHAR
;......................................................................;
;Este procedimento imprime um caracter na tela usando uma função D.O.S.;
;......................................................................;
WRITE_CHAR PROC
PUSH AX ;Põe na pilha o valor do registrador AX
MOV AH,2h ;Função 2h
INT 21h ;Interrupção 21h
POP AX ;Retira da pilha o valor inicial do registrador AX
RET ;Retorna o controle ao procedimento que chamou
WRITE_CHAR ENDP
Quinto exemplo
PRINT_ASCII PROC
MOV DL,00h ;move o valor 00h para o registrador DL
MOV CX,255 ;move o valor decimal 255 para o registrador CX
;usado para fazer um laço com 255 interações
PRINT_LOOP:
CALL WRITE_CHAR ;Chama o procedimento que imprime
INC DL ;Incrementa o valor do registrador DL
LOOP PRINT_LOOP ;Loop para imprimir 10 caracteres
MOV AH,4Ch ;Função 4Ch
INT 21h ;Interrupção 21h
PRINT_ASCII ENDP ;Finaliza o procedimento
WRITE_CHAR PROC
MOV AH,2h ;Função 2h para imprimir um caracter
INT 21h ;Imprime o caracter que está em DL
RET ;Retorna o controle ao procediemento que chamou
WRITE_CHAR ENDP ;Finaliza o procedimento
Este programa mostra na tela o valor dos 256 caracteres do código ASCII.
*****************************************************************************
CAPÍTULO 8: BIBLIOGRAFIA
Créditos:
Referências Bibliográficas:
Basic Assembler
A. Rojas
Ed Computec Editores S.A. de C.V.
Mexico
C - Caixa de Ferramentas
Carlos Augusto P. Gomes
Antonio Carlos Barbosa
Editora Érica
Página 79