Documentos de Académico
Documentos de Profesional
Documentos de Cultura
Capitulo 1a PDF
Capitulo 1a PDF
PARTE 1
Libre distribución.
Solo pongan de donde lo obtuvieron y referencias bibliográficas.
Atmel®, the Atmel logo and combinations thereof, and others are the registered trademarks or
trademarks of Atmel Corporation of its subsidiaries
Decidí hacer este tutorial por que yo aprendí a programar los Avr´s en c++ y quería
aprender en ensamblador, estuve buscando y nunca encontré un tutorial que en realidad
le entendiera entonces decidí aprender por mi cuenta. Tengo conocimientos sobre
ensamblador de Z80 y como funciona un sistema mínimo conecte uno en una protoboard
que ya no era tan sistema mínimo estaba bastante grande todavía lo tengo por ahí
armado de hecho no lo quiero desarmar, no lo uso por que que no tengo programador de
memorias paralelas. Y pues se poco ensamblador de PIC16F84 pero pues aquí tratamos
de aprender ensamblador de los Avr´s. En general el ensamblador es lo mismo en todos
los Avr´s no como en los PIC que cada uno tiene un ensamblador diferente. Desde mi
punto de vista y lo poco que conozco al arquitectura del uC Pic no fue muy bien pensada
en comparación con la de los Avr´s. Pero eso ya es decisión de cada uno.
Dedico este pequeño tutorial a todas las personas que me apoyan y me dicen que siga
adelante que cumpla con mis sueños a mis papas que siempre han sido la base de mi de
mi crecimiento como persona y que sin ellos no sabría todo esto, los quiero mucho. A mi
novia que siempre me apoya y mas que nada por el tiempo que no le dedico a ella y se lo
dedico a estas actividades que tanto me gustan.
Conocimientos Necesarios.
1. Electrónica básica(conexiones)
2. Programación ya sea en c o algún otro lenguaje.
3. Ganas de Aprender bien y hacer preguntas.
Componentes Necesarios.
Software necesario
Avr Studio 4
Avr Dude (dependiendo de el programador)
Pony Prog (dependiendo de el programador)
Sobre los componentes.
1.-Programador de Avr´s, yo tengo el Avr Dragon pero cualquiera que tengan y que
funcione debe de funcionar bien y servirnos y si no tiene chequeen paginas como
http://www.pictronico.com/shop/osc/index.php?cPath=33
que tienen desde $ 300 pesos
Aquí en Celaya Voltix vende el Avr Dragon que cuesta aproximadamente $1000
En Queretaro esta Advandix ahi venden también
En la Ciudad de México en el Centro en la calle de República de el Salvador deben de
encontrar yo e visto en al plaza con el numero 24 en la parte de arriba es cosa de que
pregunten.
O otra opción es hacerlo en lo personal yo no los e probado pero les dejo varios links.
Miren
http://www.obdev.at/products/vusb/prjprog.html
http://www.forosdeelectronica.com/f24/diagrama-programador-atmel-atmega16-32-a-
16274/
http://www.foroselectronica.es/f65/programador-atmega32-16pi-745.html
Mas adelante haber si agarramos un tutorial de estos y los probamos haber que pasa, si
alguien ve alguno y le funciono pues que nos diga
2.-Aquí en Celaya en Voltix los venden o pueden ponerse en contacto conmigo por
correo electrónico lams_007@hotmail.com y lams007@gmail.com y yo se los puedo
vender.
3.- Esta las pueden conseguir en Voltix o en Steren en cualquier tienda que venda
componentes electrónicos como para hobistas
4.-Pues esto pueden usar la de un cargador de celular una computadora o varias cosas así
o hacer una pequeña fuente con LM7805 y un transformador ahí esta el diagrama.
El Avr dude y el pony prog también son gratuitos esos hay que echarle una buscada en
google.
Desde mi punto de vista: es una computadora en un circuito integrado. Pueden tener mas
dispositivos dentro de ellos mismo como: Convertidores DAC, ADC, timers, PWM,
USART, UART, SPI, Comparadores, Interfaz usb, Ethernet, Controladores de LCD, etc..
Dependiendo de los módulos extras que tenga es su precio y complejidad. Existen
muchos fabricantes como: Atmel, Microchip, Motorola, Parallax, Hitachi, National
Instruments, Texas Instruments, Zilog , etc...
Pues bueno nosotros vamos a usar el Attiny461 pero como ya lo dije y lo vuelvo a
repetir es ensamblador y pues sirve para casi todos los microcontroladores Atmel de la
seria AVR tiny y Avr Mega no se decirles de los Xmega nunca e usado uno pero espero y
pronto poder usarlo.
Ejecución de instrucciones
Todos los uC tiene arquitectura Harvard al menos todos los que conozco esto quiere
decir tiene una memoria diferente para programa y para datos mientras que una
arquitectura Von Neuman es la misma memoria para datos y programa. Tener
arquitectura Harvard es mejor por que es mas rápido de accesar. Veamos lo así mientras
en Harvard yo lee el programa y también obtengo los datos al mismo tiempo en el
mismo ciclo de reloj en el de Von Neuman necesito dos ciclos de reloj uno para leer la
acción a hacer y la otra para el dato.
Les dejo una figura de como se lee los datos y el programa en los Avr´s
Les dejo una imagen de la arquitectura y explicación de los componentes que vamos a
usar en esta primera parte de el tutorial.
Contiene 32 Registros de propósito general de 8 bits con un tiempo de acceso de un
ciclo de reloj.
El flujo de el programa es proveído por saltos condicionados y no condicionados la
función Call o llamada a subrutina y esto es capaz de accesorio a toda la memoria de el
uC.
Statur register.(SREG)
Pues eso seria toda la teoria por este primera parte de el tutorial ahora si viene lo mismo
espero y ya tengamos al menos instalado el Avr Studio todos en nuestra maquina.
Si no recuerden descargarlo de la pagina de Atmel www.atmel.com
INICIO
Aquí le tenemos que dar click en new project y esta es la imagen que sale
Aquí seleccionamos Atmel Avr Assembler esto nos habilita el nombre de el proyecto y
el nombre de el archivo con extensión .asm y le damos en Next y sale esto.
ASEGURENSE TAMBIEN DE PONER CREATE INITIAL FILE.
Se alcanza a ver que son puras definiciones y pues así están todos en este caso buscamos la que
diga tn461def para el uC que voy a usar si tiene el Atmega8 usan la que dice mega8def y así.
Pero notamos que no esta el tn461def buscamos el uC mas parecido y pues es el Attiny2113
usamos este solo en unos caso va a haber que definir mas cosas por ejemplo este Attiny2113 no
tiene ADC y el Attiny461 si los tiene pues ya cuando los usemos vemos como le hacemos por
lo pronto no hay que preocuparnos entonces en lugar de todas esas declaraciones al inicio de le
programa ponemos
.include “tn2313def.inc”
Esto hará que todas las definiciones que tenemos en esa hoja queden en nuestro programa
cuando se este ensamblando y el ensamblador vea Sreg lo va a sustituir por el numero que
pusimos y no usamos mas memoria eso se hace a la hora de ensamblar.
Entonces tenemos algo que se ve como esto.
Ahora pues esto no hace nada incluso si se trata de ensamblar nos manda errores por que no
tiene nada este primer programa vamos a hacer que un led prenda y apague cada cierto tiempo
de hecho no se va a alcanzar a ver por que el uC lo va a hacer muy muy rápido.
El uC no sabe por donde queremos sacar el dato en esta caso hacer que se prenda o se apague el
led. Vamos a la hoja de datos y vemos tiene dos puertos de 8 bits pues que podemos hacer
sacamos todo el dato por un puerto en este caso voy a seleccionar el puerto B de alguna manera
tengo que decirle al uC que por el puerto B voy a sacar mi dato aquí es donde voy a usar mi
primera instrucción tengo que decirle al puerto B que sea salida para eso tengo que poner un
numero en el registro DDRB este numero es un 1 para que sea el bit 0 como salida pero no
puedo poner DDRB=1 tengo que cargárselo antes aun registro de propósito general para esto
vamos a hacer esto es lo que llevamos de el programación
Quiero que en el panel de salida seleccionemos el que dice PORTB lo podemos hacer mas
grande con el botón de + que tiene ahí o con seleccionarlo basta. Y donde quedamos que
estaban los archivos de el proyecto ya no están ahora hay una que dice Processor bien ahi dicen
varias cosas dice Program Counter, Stack Pointes ,....., Sreg y al final Registers hacemos mas
grande la de Registers.
Aprieten la tecla F10 para ir paso a paso en la ejecución de instrucciones o Alt+F5 vaya sola la
computadora e ligan lo que quieran y después hagan lo siguiente
Ok ya que hicieron eso necesito que se fijen en 4 cosas en el panel de salida de el uC abajo va a
salir DDRB, PINB y PORTB y vena como PORTB y PINB cambia a la hora de que pasamos
por la instrucción “OUT PORTB,R16” y ahora ne la parte de Processor quiero que vean como
el R16 cambia con las instrucciones “LDI R16,0x00” y LDI R16,0x01” y se pone con el
numero que decimos en el registro. Fíjense como la parte de hasta arriba el Program Counter va
incrementando de uno en uno hasta que llega a la instrucción RJMP y cambia a 0x0002 y si se
fijan el tan famoso Sreg nunca cambia esto quiere decir las instrucciones que estamos
ejecutando no lo afectan. Otra cosa importante a ver las instrucciones abran las instrucciones de
los Avr´s que bajaron antes y búsquenlas. Esto seria el primer programa. El siguiente es hacer
un contador que implica un poco mas de instrucciones.
Si tienen dudas sobre esto no duden en escribirme a mis dos correos lams_007@hotmail.com y
a lams007@gmail.com o dejen un comentario en www.mecatronicamexicana.blogspot.com
Esto es lo que se ve en el osciloscopio pero agregue unas instrucciones llamadas NOP para
alargar tiempo de encendido. No les posteo el otro programa por que no vale la pena.
Aquí la imagen
Les dejo también el link al video espero y les guste perdón por la cámara ya esta mal y no
enfoca bien prometo comprar una que sirva bien lo mas rápido posible.
Vean el video de youtube de este programa.
http://www.youtube.com/watch?v=39gZ2sCYfFU
Programa numero 2
Para este segundo programa omitiré todo lo que ya vimos de el primer programa. Solo que
debemos de crear un nuevo proyecto. Abordemos el problema es un Contador espero que
sepamos todos que es un contador esto quiere decir que empieza en cero y va contando hasta
que llena los 8 bits y se hace 255 aquí en este punto veremos algo interesante con el Sreg aun
no lo usaremos pero veremos algo interesante pues lo principal de nuevo usaremos el puerto b
como nuestra salida entonces lo principal sera
.include “tn2313def.inc” //definiciones
LDI R16,0xff //carga un 255 al registro 16
OUT DDRB,R16 //selecciona todo el puerto b como salida
Aquí quiero que aprendan esto si el registro DDRB tiene 0 es entrada si tiene unos es salida y si
quiero los primeros cuatro como entradas pues debo de poner los primeros cuatro en 0 y los
otros cuatro como salidas los debo de poner en uno seria este ejemplo:
De ahora en adelante lo que esta en cursiva y rojo es un ejemplo y no es parte de el programa
que estamos haciendo
include “tn2313def.inc” //definiciones
LDI R16,0xf0 //carga un 240 al registro 16
OUT DDRB,R16 //selecciona la parte alta del puerto b como salida y la baja como entrada
Espero y se allá entendido bien lo siguiente aquí seria hacer que cuente y para eso buscamos en
nuestra hoja de instrucciones y encontramos la instrucción INC que incremente un registro en 1
entonces pues si ya tengo un registro el R16 aquí vale decir que puedo usar cualquier registro
bueno no cualquiera en las hojas de instrucciones dice que registros acepta cada instrucción
entonces le agregaríamos a nuestro programa.
INC R16 // incrementa el registro 16 en 1
OUT PORTB,R16 //pone en el puerto b el valor de el R16
cabe mencionar que el R16 valía 255 antes de ejecutar INC R16 cuando se ejecuta la
instrucción cambia a 0 y aquí viene lo interesante noten que en el Sreg a la hora de compilarlo
y correrlo y se ejecuta el primer INC R16 la bandera de cero(ZF) pasa a ser uno que significa
que la ultima operación resulto ser cero y que cuando es mayor de 128 están activas las
banderas de signo y de negativo aunque esto no nos interesa ahorita. Lo único que nos faltaría
en el programa seria hacer que regrese y pues para eso tenemos la instrucción que ya
aprendimos de RJMP pero pues donde se debe de poner a donde regresa pues se pone a INC
R16 para que regrese e incremente en 1 y vuelva a sacar el dato de el puerto. Quedándonos
todo el programa como se muestra a continuación.
.include "tn2313def.inc" //definiciones
LDI R16,0xff //carga un uno al registro 16
OUT DDRB,R16 //selecciona el puerto b como salida
REGRESO: INC R16 //incremento R16 en uno
OUT PORTB,R16
NOP //esto es una nop no importa no sirve de nada ahorita
NOP //nada mas insertan tiempo muerto y gastan memoria
RJMP REGRESO
Si lo alcanzan a ver este programa es mas pequeño que el otro por que el otro había que poner
un registro a cero y aqui solo se incremente no necesitamos 2 outs.
No les pongo imagen de osciloscopio por que no puedo ver todas las señales pero pues les dejo
una foto de la protoboard
Como se ve es muy simple solo 8 leds y hacer un contador de 8bits con integrados usas muchos
mínimo 2 y este puedes hacer muchas mas cosas los leds no tiene resistencia por que son leds
que aguantan los 5 volts si tiene leds que aguanten menos voltaje si pónganles una resistencia
mínimo de 100 ohmsy máximo de 330 de ahí en mas cualquiera esta bien. Y aquí el video de
youtube. http://www.youtube.com/watch?v=wl3x5gncOPU
Sobre el video que digo que el bit mas significativo no prende tendré que investigar voy a ver
si es internamente le uC por que el programa esta bien luego alguna vez me paso que estaba
programando un Atmega32 por Isp y cuando programas por Isp un puerto de Atmega32 se
deshabilita y pensaba que no servia el puerto y leyendo encontré que no pues hay que
deshabilitar un fusible, puede estar pasando lo mismo aquí. Otra cosa que me pareció
interesante sin tener que ver con los uC se fijan como se ve morado y como rayitas eso es luz
infrarroja que detecta la cámara y el filtro no la elimina bien supongo que las rayas por que esta
espaciado es el tiempo en encendido y apagado nunca lo había notado aquí podemos aprender
de todo.
Sobre el error ya vi el error ese pin es el reset y como les dije es cosa de deshabilitar el fusible
y ya este tercer programa vamos a ver como programo físicamente el uC.
Tercer programa
Este programa va a ser un poco mas difícil por que ya va llevar toma de decisiones del uC con
el Sreg y las banderas. Va a llevar un salto y varios retardos de tiempo se podría hacer con el
Stack Pointer y la memoria Ram pero eso prefiero dejarlo para la siguiente parte de el tutorial.
Abordemos el problema pues este consiste básicamente en hacer un contador ascendente y
cuando llegue a 255 se vuelve descendente. Otro punto importante aquí si vamos a usar los 8
bits ya cheque cual era el problema como les dije un fusible con desactivarlo funciona bien y
cuando vean como lo programa pues ya con eso quedara entendido creo yo pues como siempre
empezamos igual la librería de definiciones y declarar al puerto B como salida sale.
.include "tn2313def.inc" //definiciones
LDI R16,0xff //carga un uno al registro 16
OUT DDRB,R16 //selecciona el puerto b como salida
Después de esto queremos que se incremente el R16 se detenga un tiempo para que alcanzamos
a ver como cuenta entonces tendríamos algo así
INC R16 //incremento R16
LDI R30,0x00 //cargo 0 en r30
LDI R31,0xFF //cargo 255 en R31
Aquí tenemos LDI R30,0x00 y LDI R31,0xFF esta operación ya la conocemos pero pues mas
adelante nos va a servir para hacer comparaciones recuerdan que nuestro programa va a contar
de 0, 1, 2, 3, ........, 254, 255 y después de 255, 254, 253, 252, ........, 1, 0 esos nos van a servir
para hacer eso saber cuando esta en 255 y cuando en cero.
MAS: INC R16 //muy fácil incremento R16
OUT PORTB,R16 //saco R16 por el puerto B
RJMP RETARDO //salto a retardo pero que hay en retardo
PRINCIPAL:CP R16,R31 //esto hace R16-R31 abajo lo explico
BREQ MENOS //dependiendo de las banderas salta a menos
RJMP MAS //salta a MAS que es la parte de hasta arriba de este código
Pues bueno aquí tenemos dos instrucciones nuevas CP R16,R31 y BREQ MENOS.
CP es de comparación esta hace una resta entre R16 menos R31 sin alterar el contenido de
ningún registro lo único que nos modifica son las famosas banderas o el SREG explico solo las
banderas que nos interesan
Bandera de Cero (ZF) esta se pone a uno si R16 y R31 son iguales si no es cero.
Bandera de Carry (CF)esta se pone a uno solo si R31 es mayor que R16
Aquí la bandera que vamos a usar es la de cero, no nos interesa saber si uno es mayor o menor
que otro solo si son iguales el uno a otro sale.
BREQ (Branch if equal) checa la bandera de cero si es uno salta a donde dice en nuestro
programa. Esto quiere decir que cuando R16 y R31 son iguales CP pone la bandera de cero a
uno y BREQ checa la bandera y si es uno salta a menos en el otro caso si no son iguales no
salta y continua su ejecución normal. Si recuerdan R31 contiene 0xFF esto quiere decir que
cuando llega a 255 es cuando se hace cero este ciclo es el que cuenta ascendente mente y
cuando llega a 255 salta a menos que es el que cuenta descendente mente y pues retardo se los
explico ahorita un poco mas adelante.
Cabe Recordar aqui que si quiero usar el puerto a como salida o sacar mi dato por el puerto a
en lugar de usar DDRB pongo DDRA en todo caso sera la letra de el puerto donde esta la X
DDRX, PINX y PORTX espero y esto les sirva.
Espero y les allá gustado este tutorial es para ustedes para aprender en la siguiente parte de el
tutorial vamos a ver como se usan las entradas se me ocurre un contador cada vez que yo apreté
un botón , como se usa la Ram y el satck pointer y como leer un dato analógico desde los ADC
internos que trae el micro. Manden sus dudas quejas sugerencias comentarios a
lams_007@hotmail.com y a lams007@gmail.com o dejen un comentario en
www.mecatronicamexicana.blogspot.com si les gusto o no el tutorial.
Muchas gracias por tomarse el tiempo para leer esto suerte.