Curso Fudeba de ASM


Aula 2: Algumas técnicas iniciais.

   Antes de continuar com o curso, vamos a uma breve revisão do que foi visto na aula passada. A idéia era escrever um programa, ou seja, uma seqüência de ordens para "o mais loiro dos seres" (o seu computador) que escrevesse na tela a frase "Ola Mundo".

Bem, para fazer isso, estamos usando o NotePad como editor (tem no seu Windows) e o Microsoft Macro Assembler (M80/L80 que pode ser obtido aqui:

http://www.crosswinds.net/~adrianpage/m80l80pc.zip

   Vimos algumas coisas sobre o nosso ambiente: temos 64K posições de memória, que o processador (Z80) é trabalhador, mas de memória muito curta e temos de colocar tudo que ele precisa em seus registradores (colas) antes de mandarmos ele fazer alguma coisa, e que usamos para isso o comando LoaD (LD). Vimos também como fazemos para informar o Z80 para executar uma função do DOS (STROUT), indicando para ele o numero da função (LD C,STROUT) e mandando ele ir procurar o que essa maledeta função significa no endereço BDOS (que seria o endereço do "Altavista" do MSX-DOS), pedindo para que ele voltasse e continuasse com o nosso programa quando tivesse descoberto (e executado) o que a função STROUT faz, usando o comando CALL (CALL BDOS). Finalmente vimos como voltar ao prompt do MSX-DOS, usando o comando JumP para o endereço 0 da RAM (JP 0).
   Alem disso, vimos como escrever tudo isso de forma que o M80/L80 entendam o que estamos dizendo, como colocar dados na memória (APELIDO: DB 'dado') e como dar apelidos a endereços de memória (APELIDO EQU endereço). Vimos também que o Z80 é tão "loiro" que se você mandar ele executar um dado (JP FRASE ou CALL FRASE) ele realmente vai tentar (e vai fazer caca) e por isso temos de evitar que ele chegue até estes dados (ele é como um lemingue: se você não disser para ele para onde ele tem que ir, ele vai sempre em frente, se matando no buraco, ou seja, travando o seu micro, quando for o caso... mas não sem antes jogar um monte de lixo na tela e outras coisas mais bizarras).
   Finalmente vimos como "assemblar" o programa com o M80/L80, ou seja, como tornar o que escrevemos algo legível para o computador.

   Seria interessante se o leitor pudesse adquirir o livro "MSX TopSecret" de Edison Antônio Pires de Moraes (ele andou oferecendo copias pela MSXBr-L) pois este livro contem muitas especificações que serão úteis (pra não dizer vitais) mais pra frente no curso. Programar assembly "não existe" sem informações técnicas sobre a maquina em que estamos trabalhando.
   Caso o leitor saiba inglês, pode adquirir gratuitamente o MSX2 Technical Handbook, digitado por Nestor Soriano, que contem grande parte das informações que existem no MSX TopSecret (mas nem metade delas), mas já é algum começo. Este você pode obter na pagina do KonamiMan:

http://www.geocities.com/SiliconValley/Bay/9797/msx2.htm#MSX2TH

   Ou pegue direto:

http://www.geocities.com/SiliconValley/Bay/9797/th.bin

(renomeie para th.lzh e descomprima com o LHA)

   Nesta segunda aula veremos como estender o programa que já tínhamos para fazer alguma coisa a mais, como perguntar coisas ao usuário e interpretar as respostas... vamos lá!


   Conversando com o usuário...

   Bem, para "principiar" tomemos o programa final da aula anterior.

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        LD      DE,FRASE                ; Indica posição do texto
        LD      C,STROUT                ; Indica função a ser executada pelo BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        JP      0                       ; Volta ao MSX-DOS

FRASE:          DB      'Alo, Mundo$'

        END
--- Cortar Aqui ---

   Antes de mais nada, salve este arquivo como PROG2.MAC, e então vamos fazer uma alteração substituindo o apelido FRASE pelo apelido NOMEDOPRG e substituir a frase "Alo, Mundo" por "Programa 2 - Conversando com o usuário":

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario$'

        END
--- Cortar Aqui ---

   Isso acabou de se tornar o "nome" do nosso programa. Isso vai aparecer escrito sempre que nosso programa for executado. Aconselho, inclusive, que a cada "alteração" você "assemble" o programa usando o comando CL80 PROG2 no seu PC, e leve até o MSX para verificar o que acontece, ver se funciona, o que mudou, para você perceber efetivamente o que a alteração que você fez causou. Não vou ficar escrevendo a cada instante "assemble, leve para o MSX e veja o que deu" para não aumentar ainda mais o tamanho do curso, e também para não ficar "enchendo lingüiça" com informação já conhecida.
   Bem, vamos escrever mais alguma coisa na tela, adicionando uma segunda frase chamada AUTOR, de modo que você possa colocar que foi você quem fez o programa. Isso fica assim:

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario$'
AUTOR:          DB      '  Por Daniel Caetano$'  ; coloque seu nome ao invés do meu.

        END
--- Cortar Aqui ---

   Bem, mas isso não vai fazer com que a frase apareça na tela. Precisamos mandar o Z80 mostra-la para nos. Onde mandar então? Bem, queremos que ela apareça depois dele ter mandado aparecer o nome do programa, mas antes do programa retornar ao prompt do MSX-DOS. Assim, a posição certa para inserir isso é entre o CALL BDOS e o JP 0. Nosso programa, com as alterações fica assim:

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario$'
AUTOR:          DB      '  Por Daniel Caetano$'  ; coloque seu nome ao invés do meu.

        END
--- Cortar Aqui ---

   Se você observar bem, a coisa já começou a ficar meio bagunçada. Assim é bom inserir alguns comentários chamados "comentários de blocos", que são comentários que descrevem não o que um comando faz, mas o que um conjunto de comandos, logo abaixo deste comentário, fazem. Nos temos atualmente dois conjuntos básicos de comandos: um que escreve o nome do programa e outro que escreve o nome do autor. Vamos inserir esses comentários: face="Comic Sans MS, Arial, Helvetica, Sans-Serif" size="2">

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario$'
AUTOR:          DB      '  Por Daniel Caetano$'  ; coloque seu nome ao invés do meu.

        END
--- Cortar Aqui ---

   Agora isso não parece tão interessante, mas no futuro você verá que isso é o que torna os programas "legíveis" depois de muito tempo que você não os vê. Se você executar esse programa, vai ver que algo "indesejado" aconteceu: o MSX-DOS *não* pulou linha entre uma frase e outra. Bem, pois é, você precisa dizer pra ele fazer isso. Pra isso você precisa dizer para ele retornar para a primeira posição da linha (CR, de Carriage Return) e pular uma linha (LF, de Line Feed). O CR é representado pelo numero 13, e o LF pelo numero 10. Assim, se você quiser pular linha, basta adicionar, no fim da frase, esses números, da seguinte forma:

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario$',13,10

   MAS, se você fizer isso verá que não adiantou nada. O programa continua *não* pulando a linha. Uai... E por que não? Simples... Se você observar, tem um "$" antes do 13 e do 10, sendo que o $ diz para a função STROUT que o texto ACABOU. Então, meu amigo, esse 13 e esse 10 que você colocou ai' não estão nem sendo lidos. Você precisa coloca-los ANTES do $, o que pode ser feito assim:

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario',13,10,'$'

   E agora sim ele vai pular as linhas direitinho. Nosso programa, já corrigido, fica assim:

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuario',13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,'$'

        END
--- Cortar Aqui ---

   Agora vamos pedir ao usuário que pressione uma tecla, receber a resposta e fazer alguma coisa com essa resposta.
   O primeiro passo é colocar a pergunta na tela, para que o usuário tome conhecimento de que precisa entrar com alguma informação. Isso pode ser feito da mesma forma com que fizemos antes com o nome do programa e nome do autor, inserindo a frase PERGU1:

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'

        END
--- Cortar Aqui ---

   Legal, nosso programa, então, já mostra suas informações e faz uma pergunta. Bem, você deve ter notado que a saída do programa foi algo mais ou menos assim:

A>PROG2
Programa 2 - Conversando com o usuário
  Por Daniel Caetano
  Pressione alguma tecla:
A>

   Mas seria mais interessante se a saída fosse:

A>PROG2
Programa 2 - Conversando com o usuário
  Por Daniel Caetano

  Pressione alguma tecla:
A>

   Ficaria mais legal, não? Pois é. Isso é simples. Basta mandar pular mais uma linha, acrescentando um 10 adicional na frase:

AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'

   Com isso alteramos o programa assim:

--- Cortar Aqui ---
STROUT  EQU     9
BDOS    EQU     5

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'

        END
--- Cortar Aqui ---

   Mas e agora? O programa já mostra tudo como a gente quer, mas como informar o Z80 para ele receber uma informação? Bem, existe uma função do MSX-DOS especifica para que possamos esperar que uma tecla seja pressionada. Essa função chama-se CONIN (CONsole IN, entrada de console, o que é o mesmo que entrada pelo teclado), e é a função de numero 1 do BDOS. Para chama-la, basta indicar a função, uma vez que nada mais é necessário para que o Z80 possa executar a operação. Isso pode ser feito assim:

        LD      C,1                     ; Indica a função que pega uma tecla
        CALL    5                       ; Chama BDOS

   Ou, usando os apelidos,

        CONIN   EQU     1
        BDOS    EQU     5
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

   Bem, podemos inserir isso logo abaixo da pergunta, em nosso programa:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'

        END
--- Cortar Aqui ---

   Bem, tudo jóia, mas quando executei isso tudo que aconteceu foi que o programa esperou eu pressionar uma tecla, mostrou a tecla que eu pressionei e voltou para o prompt. Note inclusive que o prompt apareceu exatamente na frente da pergunta... Se você observar no programa, isso ocorreu porque *não* mandamos pular linha nem voltar ao inicio da linha na frase PERGU1. Legal, mas como faço para saber, dentro do meu programa, qual foi a tecla que foi pressionada?
   O Z80 é um cara legal. Tem memória curta, mas é legal. Ele compartilha as colas dele com a gente. Existe uma cola especial que é normalmente onde ele coloca as respostas das coisas que perguntamos. Esse registrador (essa cola) é o chamado A. Por exemplo, usando a função CONIN do BDOS, perguntamos a ele que espere por uma tecla ser pressionada, e quando ela for, que nos informe qual foi. Assim, ele coloca o numero dessa tecla (aquele mesmo numero "ASCII" que eu citei na aula anterior) no registrador A. Então, já temos a resposta. A tecla digitada está prontinha para usarmos, no registrador A.
   Interessante, mas isso parece ainda não ajudar muito, não? Bem, antes de ver como podemos usar essa informação dada pelo Z80, vamos preparar uma resposta ao usuário. Vamos inserir a seguinte frase:

RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'

   Note que antes da frase eu devidamente adicionei o retorno do cursor (13 = CR) e pulei duas linhas (10 = LF). Alem disso, não pulei linha nem retornei o cursor ao fim da frase, pois vou querer colocar a resposta (a tecla pressionada) DEPOIS dela, sem pular linha ou qualquer outra coisa. Adicionamos essa frase depois do ponto em que mandamos o Z80 ir pegar uma tecla pra gente:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

        ; Mostra a resposta
        LD      DE,RESP1                ; Indica texto da resposta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.


        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'

        END
--- Cortar Aqui ---

   Bom, o programa já quase faz o que a gente quer, mas ainda não faz. Eu tenho o valor da tecla, mas não sei como mostrar isso na tela. Ou sei? Sei sim. Existe no BDOS uma função que faz exatamente o contrario do que a CONIN faz. Não, essa função não escreve um caractere no teclado. Essa função escreve um caractere na tela, de acordo com o numero fornecido. Essa função chama-se CONOUT (CONsole OUT, ou saída para o console, ou saída para a tela - entenda, o console é um negocio que tem uma entrada (o teclado) e uma saída (a tela). Assim, sempre que você pedir uma ENTRADA (IN) do console, o computador lê o teclado. Sempre que você pedir uma SAIDA (OUT) para o console, o computador escreve na tela).

   Essa função é chamada normalmente, da mesma forma que a CONIN, pois é também uma função do MSX-DOS. A função CONOUT é a função de numero 2 do MSX-DOS. Há um porem... Ela requer que o numero do caractere a ser escrito esteja em uma outra cola... No registrador chamado E. Bem, temos o numero do caractere no registrador A, e precisamos que ele esteja no E. Existem diversas maneiras de copiar o valor de um registrador para outro, mas o mais fácil é usando o LoaD (LD). Se você fosse mandar alguém copiar a cola, você talvez dissesse:

   Copie na cola E o que estiver na cola A

   Para o Z80 é a mesma coisa:

         LD      E,A

   E pronto... após isso, o valor do caractere vai estar onde queremos, no registrador E. Note que o comando LoaD (LD) COPIA o valor de um registrador para outro, portanto, o que quer que estivesse no registrador E foi SUBSTITUIDO pelo valor do registrador A. Alem disso, como se trata de uma COPIA, o valor que estava no registrador A (o numero do caractere) CONTINUA lá.
   Inserindo essa nova informação no nosso programa, temos:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

        ; Mostra a resposta
        LD      DE,RESP1                ; Indica texto da resposta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        LD      E,A                     ; Copia valor do caractere para E

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'

        END
--- Cortar Aqui ---

   Mas só isso não adianta, né? Precisamos, alem disso, usar nosso conhecimento da função CONOUT para mostrar o caractere. Para tanto vamos inserir mais isso no programa:

        CONOUT  EQU     2

        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

   Ficando nosso programa assim:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
CONOUT  EQU     2
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

        ; Mostra a resposta
        LD      DE,RESP1                ; Indica texto da resposta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        LD      E,A                     ; Copia valor do caractere para E
        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'

        END
--- Cortar Aqui ---

   Beleza? Bem, se você perdeu tempo para executar esse programa, você vai ver que, sinistramente, algo deu errado. Esse programa sempre vai dizer que a tecla pressionada foi a tecla $. E porque isso acontece?
   Será que tem um bug...? COM CERTEZA, e um bug conceitual, daqueles mais difíceis de se encontrar. Como vamos encontra-lo? Bem, fizemos algumas coisas entre o momento em que obtivemos o valor do caractere em A e o momento em que copiamos ele para E. Vamos ver se, eliminando esse monte de coisa, a saída do programa acontece normalmente. Vamos "comentar" as linhas adicionando um ";" na frente delas, ficando nosso programa assim:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
CONOUT  EQU     2
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS

        ; Mostra a resposta
;       LD      DE,RESP1                ; Indica texto da resposta
;       LD      C,STROUT                ; Indica função de mostrar texto do BDOS
;       CALL    BDOS                    ; Manda o BDOS executar.
        LD      E,A                     ; Copia valor do caractere para E
        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'

        END
--- Cortar Aqui ---

   Ou seja, eliminamos tudo que acontece entre o momento em que pegamos o numero do caractere e o momento em que mandamos ele ser mostrado na tela. Rodemos o programa e ... veja que funciona! Note que agora o caractere aparece repetido duas vezes na tela (uma vez é o "retorno" normal da função CONIN e a outra é a saída da função CONOUT). Então podemos presumir que estamos pegando e mostrando o caractere corretamente. O que pode ter dado errado então? Bem, se você observar, verá que entre pegarmos o valor da tecla pressionada e tentarmos imprimir ele na tela, imprimimos um texto: " A tecla pressionada foi: "... O problema deve estar ai'.
   Lembra-se de que na primeira aula eu falei que o Z80 tinha poucos papeis de cola e que portanto ele precisava reciclar esses papeis? Provavelmente é o que está acontecendo. Em algum momento, quando o MSX-DOS está imprimindo a sua frase, ele usa o registrador A para guardar o numero do caractere a ser impresso na tela. Um indicio de que isso é verdade é que, se você observar, o valor que era sempre impresso como sendo "a tecla que você digitou" era o "$", que é exatamente o ultimo caractere da frase RESP1!
   Bem, então como fazemos? Será que a gente não pode nunca fazer nada entre pegar um dado e usa-lo? Claro que podemos. Se não pudéssemos, seria impossível trabalhar em Assembly. O que precisamos é guardar erra valor na MEMORIA, em um lugar seguro, antes de fazermos qualquer outra coisa. Existem muitos jeitos de guardar este dado na memória. Um é dizendo exatamente onde queremos que este dado seja guardado. Outro é deixando que o Z80 escolha para a gente. Neste momento vamos escolher a primeira opção. Então, primeiro vamos definir um lugar para esse valor ser colocado. Um bom lugar para isso é depois das frases. E como guardamos lugar para esse caractere? Simples, definimos uma "frase" para ele, de um caractere só. Isso pode ser feito assim:

 VAR1:   DB      0

   Isso define uma posição de memória apelidada de VAR1 (VARiavel 1) com o valor 0 dentro. É neste lugar que vamos mandar o Z80 colocar o valor que a gente recebe. E como vamos fazer isso? Exatamente com o comando LoaD. Ele também serve para copiar coisas da MEMORIA para um REGISTRADOR, ou de um REGISTRADOR para a MEMORIA. Se queremos colocar o dado em VAR1 e este dado está no registrador A, nada mais natural do que um comando:

         LD      VAR1,A

   Mas isso não vai funcionar, como diria o DalPoz. E porque? Bem, lembre-se que VAR1 é apenas uma apelido para um NOME de posição de memória. O Z80 iria ler isso ai' como "De um jeito de o nome 0 ser igual a 2", por exemplo. É claro que teremos problemas. Temos que dar um jeito de dizer a ele que coloque o valor de A DENTRO da posição de memória chamada VAR1. Para isso usamos os parênteses:

        LD      (VAR1),A

   Isso diz ao Z80 "Copie DENTRO da posição de memória chamada VAR1 o valor de A". Ai' sim obteremos sucesso, pois ele vai entender direitinho o que queremos dizer... o que em nosso programa fica:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
CONOUT  EQU     2
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS
        LD      (VAR1),A                ; Grava o valor da tecla na MEMORIA.

        ; Mostra a resposta
;       LD      DE,RESP1                ; Indica texto da resposta
;       LD      C,STROUT                ; Indica função de mostrar texto do BDOS
;       CALL    BDOS                    ; Manda o BDOS executar.
        LD      E,A                     ; Copia valor do caractere para E
        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'
VAR1:           DB      0

        END
--- Cortar Aqui ---

   Ótimo, o dado já está na memória. Agora temos que, antes de chamar a função que nos mostra o caractere na tela, colocar no registrador E esse valor que está na posição de memória VAR1. O jeito mais natural de fazer isso é:

        LD      E,(VAR1)

   Ou seja, "Copie em E o que estiver DENTRO da posição de memória chamada VAR1". Substituindo a antiga instrução LD E,A (que tinha exatamente a função disso que estamos fazendo) por essa nova instrução LD E,(VAR1) e retirando os ";" das linhas que mostram o texto da resposta, teremos:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
CONOUT  EQU     2
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS
        LD      (VAR1),A                ; Grava o valor da tecla na MEMORIA.

        ; Mostra a resposta
        LD      DE,RESP1                ; Indica texto da resposta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        LD      E,(VAR1)                ; Copia para E o conteúdo de VAR1.
        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'
VAR1:           DB      0

        END
--- Cortar Aqui ---

   Parece que agora vai, não? Bem, ao tentar compilar você seguramente vai receber a seguinte mensagem:

C:\ASMMSX>cl80 prog2
M80/L80 Z80 Compiler - IBM PC
Ported by A&L Software
ÿ
MSX.M-80  1.00  01-Apr-85  (c) 1981,1985 Microsoft
A 0028'   3A 00AÉ              LD      E,(VAR1)                ; Copia para E o
 conteúdo de VAR1.

1 Fatal error(s)

C:\ASMMSX>

   E o seu programa não terá nem sido assemblado. O que esse código de erro significa? Significa que na linha apresentada ai' houve um erro. E no caso O erro é que esse comando LD E,(VAR1) não existe para o Z80!
   Caramba, você me pergunta, mas é tão lógico! Como assim, não existe? Simples, não existindo. E essa é a parte mais chata do Assembly. Lembra que eu disse que existiam colas especificas para algumas coisas? Pois é. Em um outro momento eu disse que o Z80 gostava de colocar repostas no registrador A. Pois aqui é exatamente um desses casos. Você só pode ler o conteúdo de UM BYTE da MEMORIA, ou seja, o valor de UMA única posição de memória *se* coloca-lo dentro do registrador A.
   Você vai dizer... poxa, mas eu preciso dele dentro do registrador E, e não dentro do registrador A. Felizmente você ainda pode usar o LoaD para copiar o valor de A para E. Assim, o comando:

        LD      E,(VAR1)                ; Copia para E o conteúdo de VAR1.

   Deve ser substituído por dois outros comandos, que tem um resultado final equivalente:

        LD      A,(VAR1)                ; Copia para A o conteúdo de VAR1
        LD      E,A                     ; Copia para E o conteúdo de A

   E o nosso programa finalmente fica assim:

--- Cortar Aqui ---
BDOS    EQU     5

CONIN   EQU     1
CONOUT  EQU     2
STROUT  EQU     9

START:
        ; Mostra nome do programa
        LD      DE,NOMEDOPRG            ; Indica texto do nome do programa
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra nome do programador
        LD      DE,AUTOR                ; Indica texto do nome do autor
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Mostra a pergunta
        LD      DE,PERGU1               ; Indica texto da pergunta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.

        ; Espera tecla ser pressionada e coloca seu valor em A
        LD      C,CONIN                 ; Indica a função que pega uma tecla
        CALL    BDOS                    ; Chama BDOS
        LD      (VAR1),A                ; Grava o valor da tecla na MEMORIA.

        ; Mostra a resposta
        LD      DE,RESP1                ; Indica texto da resposta
        LD      C,STROUT                ; Indica função de mostrar texto do BDOS
        CALL    BDOS                    ; Manda o BDOS executar.
        LD      A,(VAR1)                ; Copia para A o conteúdo de VAR1
        LD      E,A                     ; Copia para E o conteúdo de A
        LD      C,CONOUT                ; Indica a função exibe um caractere na tela
        CALL    BDOS                    ; Chama BDOS

        JP      0                       ; Volta ao MSX-DOS

NOMEDOPRG:      DB      'Programa 2 - Conversando com o usuarió,13,10,'$'
AUTOR:          DB      '  Por Daniel Caetano',13,10,10,'$'
PERGU1:         DB      '  Pressione alguma tecla: $'
RESP1:          DB      13,10,10,'  A tecla pressionada foi: $'
VAR1:           DB      0

        END
--- Cortar Aqui ---

   Finalmente nosso programa está pronto e faz o que queremos. Nesta aula aprendemos novos comandos, e vimos como algumas limitações do Assembly do Z80 nos "enchem a paciência", mas teremos de conviver com elas se quisermos programas nesta linguagem. Adianto que isso que parece uma irritação agora no futuro não lhes incomodará mais, passando a ser algo "natural".
   Também vimos algumas novas funções, como a de pegar um caractere do teclado e a de mostrar um caractere na tela.
   Na próxima aula veremos como "otimizar" este programa e também como usar o BrMSX para verificar o resultado da programação, para evitar termos de ficar copiando as coisas em disquete e levando até o MSX toda hora... Vocês verão que isso é de grande auxilio e acelera muito a velocidade do desenvolvimento.

   Espero que tenham gostado. Quaisquer duvidas, enviem para a lista, eu ou qualquer um dos programadores da lista responderão tão breve quanto possível.

Abraços,

Daniel Caetano.

PS: Sugestões e comentários são bem-vindos. (^=



Ufa!
Em Construção