TUTORIAL DE ATMEL AVR´s de 8-bits Logo propiedad de Atmel Ensamblador Autor: Luis Antonio Méndez Soto 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 "AVR®" and "AVR is a registered trademark of Atmel Corporation". Introducción. 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. 1.-Programador de Avr´s cualquiera que funcione. 2.-uC Atmel Serie Avr Attiny 461, Atmega8(cualquier Avr funciona por que va a ser en ensamblador y todo ensamblador de Avr es muy parecido) 3.-Protoboard 4.-Pequeña fuente de 5 Volts 5.-Resistencias (variadas) 6.-Led´s Varios 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-a16274/ 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. 5.- Igualmente en Voltix o en cualquier Steren si quiere me pueden contactar y yo también se las puedo vender. 6.- Igualmente en Voltix o en cualquier Steren si quiere me pueden contactar y yo también se las puedo vender. Software necesario Pues el Avr Studio lo pueden conseguir directo de la pagina de Atmel www.atmel.com este software es gratuito solo hace falta registrarse y si quieren aprender pues con este basta por que también es simulador y para programas simples como con los que vamos a empezar con este software basta. El Avr dude y el pony prog también son gratuitos esos hay que echarle una buscada en google. Les recomiendo también tener la hoja de datos de su uC que quieren programar la pueden buscar directo en Atmel tiene todo muy bien organizado y las instrucciones de ensamblador de loa Avr´s también están en Atmel pero a esa si les dejo el link seria http://www.atmel.com/dyn/resources/prod_documents/doc0856.pdf Estas son las instrucciones de los Avr´s de 8 bits Inicio y pura teoria. Que es un Microcontrolador?De Wikipedia: Un circuito Integrado o chip que incluye en sus interior las tres unidades básicas de una computadora: Unidad central de procesamiento (CPU) , Memoria y unidades de E/S. 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... Aplicaciones: Pues sus aplicaciones son prácticamente infinitas, imaginate cuantos aparatos conoces que tiene una pantalla ya sea de leds, touch, lcd como hornos, celulares,cámaras, refrigeradores, relojes, aires acondicionados, lavadoras, controles remotos, etc... Los aparatos que traen botones para interactuar con nosotros, los aparatos con leds pequeños y así muchos mas. Todo este tipo de aparatos necesita un controlador, generalmente es una computadora pero son muy caras para tenerlas dentro, ahí es donde entra el uC. Un carro es un buen ejemplo, sin temor a equivocarme un carro moderno trae cerca de 50 uC, casi casi uno por sensor por que van interconectados a un una computadora central que controla todo pero algo necesita enviar los datos de una manera para que los entienda el CPU principal esto lo hace el uC. El bus que usan generalmente los carros se llama Bus CAN. 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 Imagen propiedad de Atmel Figura 1 de la hoja de datos del Attiny461 Como se alcanza a ver en el primer ciclo de reloj se decodifica la primera instrucción y en el segundo ciclo ya se sabe que tiene que hacer y ejecuta la primera instrucción, mientras que en ese mismo ciclo la segunda instrucción se decodifica y al tercer ciclo se ejecuta esa segunda instrucción y se obtiene la 3er instrucción y así sucesivamente, todo esto es gracias a las dos memorias separadas para datos y para programa. Un poco de la arquitectura. 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. Imagen propiedad de Atmel ALU Este esta conectado a los 32 registros de propósito general.se ejecutan instrucciones entre los registros en un solo ciclo de reloj puede ser con otro registro o con otro dato. Se divide en tres categorías :operaciones aritméticas, operaciones lógicas y operaciones de tipo bit. Statur register.(SREG) Contiene información de la operación aritmética ejecutada mas recientemente. Este registro no se guarda automáticamente cuando entras a una subrutina, hay que guardarlo automáticamente. Imagen propiedad de Atmel Esta parte es muy importante es la que nos dice como debe de seguir el programa o las condiciones apréndansela o impriman la es muy importante les explico cada bit. Bit 7 Global Interupt enable (GIE) Cuando este bit es 1 están habilitadas la interrupciones del uC, si no esta habilitado y en los otros registros se habilitan no va a pasar nada por que este no esta habilitado. Se cambia a 0 automáticamente por hardware cuando ocurre alguna interrupción. Y puede ser puesto a uno de nuevo con la instrucción RETI. Puede ser puesto a 1 o 0 con las instrucciones SEI y CLI. Bit 6 Bit Copy Storage (BCS) Este se usa para copiar un bit de el SREG a un registro o de un registro al SREG no se usa mucho en programación. Bit 5 Half Carry Flag (HCF) Indica si hay medio carry en la ultima operación. O si la operación anterior fue mayor a 15. Bit 4 Sign Bit (SB) Siempre es una operación exclusive or con la bandera de negativo y la de complemento a dos y sobre flujo Bit 3 Two complements overflow flag Esta la Explico un poco mas adelante Bit 2 Negative flag (NF) Indica si el resultado de la ultima operación fue menor que cero. Tambien podemos checarla y ver que el numero es 128 o mayor Bit 1 Zero flag(ZF) Indica si la ultima operación fue cero. Bit 0 Carry flag (CF) Indica si el resultado de la ultima operación fue mas de 255. Sobre los registros. Los registros son una especie de memoria Ram pero no lo son estos están conectados directo al ALU y todas las operaciones pasan por el ALU. Para un programa pequeño no necesita Ram pero si necesita guardar información para esto nos sirven los registros como lo dice son 32 registros les dejo una pequeña imagen de como están. Imagen propiedad de Atmel Como se alcanza a ver aquí los registro del 26 al 31 de pueden usar como registros de 16 bits para lograr accesar a toda la memoria del uC. Las operaciones que se pueden hacer son: Una entrada de 8 bits de entrada y el resultado ser de 8 bits. Dos entradas de 8 bits y un resultado de 8 bits. Dos entradas de 8 bits y una salida de 16 bits. Una entrada de 16 bits y una salida de 16 bits. 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 Esta es la imagen con la que empezamos 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. Aquí seleccionamos Avr Simulator y en mi caso Selecciono Attiny461 si ustedes tiene otro uC seleccionen el modelo de el uC que tiene si no aparece en Avr Simulator seleccionen el Avr Simulator 2 y ahí de seguro debe aparecer y damos click en Finish. Pues ahora si viene lo bueno no se si recuerden el Sreg que tiene un numero en hexadecimal a la hora de programar no existe el Sreg y los demás registros del timer, adc usart y demás cosas cada uno tiene un numero pero pues así es un poco mas difícil recordarlo por ejemplo el que les digo el Sreg tiene el numero hexadecimal 0x3F de ahora en adelante cuando vean 0xA0 va a ser un numero en hexadecimal es mas fácil aprender que es el Sreg que el registro 0x3F. Entonces lo que podemos hacer es que escribamos. De ahora en adelante todo texto en cursiva y negritas es parte de el programa .equ Sreg = 0x3F y así para todos los demás registros que vayamos a y usar. Pero esta la otra opción que se me hace mas fácil, donde se instalo el Avr Studio en mi caso C:\Program Files\Atmel hay una carpeta que dice Avr Tools le damos click ahí después hay una que dice Avr Assembler click y por fin en la que dice Appnotes. Vemos que nos aparece cosas así : m32def, m48def, tn2313def, tn45def, etc... abramos uno el que sea en este caso les pego una parte de lo que tiene el tn2313def aquí lo tienen: ;*************************************************************************** ;* A P P L I C A T I O N N O T E F O R T H E A V R F A M I L Y ;* Number :AVR000 ;* File Name :"tn2313def.inc" ;* Title :Register/Bit Definitions for the ATtiny2313 ;* Date :03.06.17 ;* Version :1.00 ;* Support E-mail :avr@atmel.com ;* Target MCU :ATtiny2313 ;* DESCRIPTION ;* When including this file in the assembly program file, all I/O register names and I/O register bit names appearing in the data book can be used.In addition, the two registers forming the data pointer Z have been assigned names ZL - ZH. ;* The Register names are represented by their hexadecimal address. ;* The Register Bit names are represented by their bit number (0-7). ;* Please observe the difference in using the bit names with instructions such as "sbr"/"cbr" (set/clear bit in register) and "sbrs"/"sbrc" (skip if bit in register set/cleared). The following example illustrates this: ;* in r16,PORTB ;read PORTB latch ;* sbr r16,(1<<PB6)+(1<<PB5) ;set PB6 and PB5 (use masks, not bit#) ;* out PORTB,r16 ;output to PORTB ;* in r16,TIFR ;read the Timer Interrupt Flag Register ;* sbrc r16,TOV0 ;test the overflow flag (use bit#) ;* rjmp TOV0_is_set ;jump if set ;* ... ;otherwise do something else ;*************************************************************************** ;***** Specify Device .device ATtiny2313 ;***************************************************************************** ; I/O Register Definitions ;***************************************************************************** .equ SREG = 0x3F .equ SPL = 0x3D .equ OCR0B = 0x3C .equ GIMSK = 0x3B .equ EIFR = 0x3A .equ GIFR = 0x3A ; for compatibility purpose .equ TIMSK = 0x39 .equ TIFR = 0x38 .equ SPMCSR = 0x37 .equ OCR0A = 0x36 .equ MCUCR = 0x35 .equ MCUSR = 0x34 .equ TCCR0B = 0x33 .equ TCCR0 = 0x33 ; for compatibility purpose .equ TCNT0 = 0x32 .equ OSCCAL = 0x31 .equ TCCR0A = 0x30 .equ TCCR1A = 0x2F .equ TCCR1B = 0x2E .equ TCNT1H = 0x2D .equ TCNT1L = 0x2C .equ OCR1AH = 0x2B .equ OCR1AL = 0x2A .equ OCR1BH = 0x29 .equ OCR1BL = 0x28 .equ CLKPR = 0x26 .equ ICR1H = 0x25 .equ ICR1L = 0x24 .equ GTCCR = 0x23 .equ SFIOR = 0x23 ; for compatibility purpose .equ TCCR1C = 0x22 .equ WDTCR = 0x21 .equ PCMSK = 0x20 .equ EEAR = 0x1E ; for compatibility purpose .equ EEARL = 0x1E .equ EEDR = 0x1D .equ EECR = 0x1C .equ PORTA = 0x1B .equ DDRA = 0x1A .equ PINA = 0x19 .equ PORTB = 0x18 .equ DDRB = 0x17 .equ PINB = 0x16 .equ GPIOR2 = 0x15 .equ GPIOR1 = 0x14 .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ .equ GPIOR0 = 0x13 PORTD = 0x12 DDRD = 0x11 PIND = 0x10 USIDR = 0x0F USISR = 0x0E USICR = 0x0D UDR = 0x0C UCSRA = 0x0B USR = 0x0B ; for compatibility purpose UCSRB = 0x0A UCR = 0x0A ; for compatibility purpose UBRRL = 0x09 UBRR = 0x09 ; for compatibility purpose ACSR = 0x08 UCSRC = 0x03 UBRRH = 0x02 DIDR = 0x01 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 .include “tn2313def.inc” //definiciones LDI R16,0x01 //carga un uno al registro 16 OUT DDRB,R16 //selecciona el bit0 del puerto b como salida Bien ahora no falta decirle que se prenda y se apague cada X tiempo para eso volvemos a usar la instrucción OUT pero en lugar de DDRB que dice si el puerto es entrada o salida ponemos PORTB que dice cuanto va a valer el PORTB quedando OUT PORTB,R16 //aqui saco el valor de el R16 por el puertoB Bien ahora solo falta que se haga o para ver que si funciona entonces seria poner en un registro un cero y usar la otra vez la instrucción out quedando de esta manera. LDI R16,0x00 //uso de nuevo el R16 para que este un poco tiempo prendido OUT PORTB,R6 //saco el cero que tiene el registro r16 Esto seria en si nuestro programa básico miren .include “tn2313def.inc” //definiciones LDI R16,0x01 //carga un uno al registro 16 OUT DDRB,R16 //selecciona el bit0 del puerto b como salida OUT PORTB,R16 //aqui saco el valor de el R16 por el puertoB LDI R16,0x00 //uso de nuevo el R16 para que este un poco tiempo prendido OUT PORTB,R6 //saco el cero que tiene el registro r16 Pero nuestro aparato no se puede decir que es inteligente por que queremos que lo haga por siempre y si lo dejamos así lo va a hacer una vez y ya no va a hacer mas entonces hay que encontrar una manera para que lo repita por siempre y para eso usamos la función RJMP que brinca a una parte de el programa que queramos nosotros. El problema aquí es decidir a donde regresar podríamos regresar a la primera linea la de LDI R16,0x01y quedaría bien pero pues en cada nuevo ciclo declararíamos otra vez que el bit0 de el puerto B es salida entonces yo mejor agrego una linea de que diga LDI R16,0x01 y después el RJMP quedando todo el programa como: .include “tn2313def.inc” //definiciones LDI R16,0x01 //carga un uno al registro 16 OUT DDRB,R16 //selecciona el bit0 del puerto b como salida INICIO: OUT PORTB,R16 //aqui saco el valor de el R16 por el puertoB LDI R16,0x00 //uso de nuevo el R16 para que este un poco tiempo prendido OUT PORTB,R6 //saco el cero que tiene el registro r16 LDI R16,0x01 //cargo de nuevo un uno en el R16 RJMP INICIO // salto a inicio Después de RJMP pongo a donde quiero saltar puedo usar cualquier palabra o dirección de memoria pero es mas fácil la palabra aquí tendríamos un led apagándose y prendiendo. Solo falta compilar y correr el programa para esto podemos usar la tecla F7 o usar en la barra de herramientas donde dice build darla “build” o “build and run” la diferencia es que build solo lo compila y build and run lo compila y corre usemos build and run para ver que pasa ok En al parte de los errores no debe de aparecer esto para decir que todo nuestro programa esta bien. Quedamos que le ibamos a dar en build and run para simularlo entonces el programa ya debe de estar corriendo esperando a que le digamos que ejecute la siguiente operacion debe de estar algo asi. 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. Aquí otra parte de el código MENOS: DEC R16 //no hay mucho que decir decremento en uno R16 OUT PORTB,R16 RJMP RETARDO1 //Salta a Retardo1 ahorita vemos que es retardo1 PRINCIPAL1:CP R16,R30 //comparo R16 con R30 BREQ MAS //dependiendo de las banderas salta a mas RJMP MENOS //Salta a menos Pues primero voy a empezar a omitir la partes que ya vimos mas bien no las voy a explicar. Aquí tenemos una nueva instrucción DEC es muy fácil es como INC pero en lugar de sumar uno resta uno. Aquí se fijan tengo RJMP RETARDO1 es el mismo código que RETARDO sin el 1 pero como el Avr no recuerda en donde estaba por eso dos rutinas de retardo si usáramos la función CALL y el Stack Pointer (SP) no necesitaremos dos pero eso sera para la siguiente parte de el tutorial. Y pues para lo demas seria los mismo que el otro Resto R16-R30 si el resultado es cero salta a MAS si no salta a menos. Ahora si el tan famoso retardo. RETARDO:LDI R25,0x00 REGRESO:INC R25 CP R25,R31 //aquí comparo R25 con R31 deben de ser 255 BREQ ACABO //si son iguales se va acabo y de acabo salta a RJMP REGRESO //principal que es la parte que cuenta hacia arriba ACABO: RJMP PRINCIPAL //si no a acabado salta a regreso y sigue contando RETARDO1:LDI R25,0x00 REGRESO1:INC R25 CP R25,R31 BREQ ACABO1 // es lo mismo pero para la parte de cuenta hacia RJMP REGRESO1 //abajo espero y lo entiendan ACABO1: RJMP PRINCIPAL1 Espero y se entienda es muy parecido al contador aquí si yo quiero mas tiempo puedo agregar NOP entre INC R25 y BREQ ACABO aquí el ejemplo: REGRESO1:INC R25 NOP //Inserto tiempo muerto NOP //Inserto tiempo muerto CP R25,R31 BREQ ACABO1 // es lo mismo pero para la parte de cuenta hacia RJMP REGRESO1 //abajo espero y lo entiendan ACABO1: RJMP PRINCIPAL1 Para aproximar el tiempo de retardo aquí hay que ver dos cosas el lazo o el ciclo que se hace que vendría siendo todo lo que esta entre INC R y RJMP REGRESO vemos las instrucciones que seria en este caso 6 y vemos cuantos ciclos de reloj toma cada una aquí todas toman una menos RJMP que lleva dos y BREQ cuando salta también son dos esa la podemos agregar al final por que solo salta una vez. Y pues se hacen 255 veces esas siete por que cuenta hasta 255 que serian 1785 instrucciones Y debemos agregarle dos ciclos de reloj por el RJMP REGRESO, RJMP PRINCIPAL1 y el ciclo que nos faltaba serian 5 ciclos mas entonces tomamos 1790 ciclos. Suponiendo que nuestro uC corre a 8MHZ eso hace que haga 8 millones de ciclos por segundo entonces hacemos la simple regla de tres 8 millones es a un segundo como 1790 es a cuanto que serian 0.00022375 s que es un retardo muy pequeño lo que podemos hacer es añadir mas NOP o podemos poner un ciclo dentro de esos lo cual nos daría un retardo mucho mas grande. Aquí les dejo tal cual el programa que yo le puse al uC de el video. .include "tn2313def.inc" //definiciones LDI R16,0xff //carga un uno al registro 16 OUT DDRB,R16 //selecciona el puerto b como salida INC R16 LDI R30,0x00 LDI R31,0xFF MAS: INC R16 OUT PORTB,R16 RJMP RETARDO PRINCIPAL:CP R16,R31 BREQ MENOS RJMP MAS MENOS: DEC R16 OUT PORTB,R16 RJMP RETARDO1 PRINCIPAL1:CP R16,R30 BREQ MAS RJMP MENOS RETARDO:LDI R25,0x00 REGRESO:INC R25 NOP NOP NOP NOP NOP NOP NOP NOP NOP NOP CP R25,R31 BREQ ACABO RJMP REGRESO ACABO: RJMP PRINCIPAL RETARDO1:LDI R25,0x00 REGRESO1:INC R25 NOP NOP NOP NOP NOP NOP NOP NOP NOP NOP CP R25,R31 BREQ ACABO1 RJMP REGRESO1 ACABO1: RJMP PRINCIPAL1 Como ven es mucho mas grande por las NOP y tiene un retardo no muy grande de 0.00047875 s o 4.7875 mili segundos. Les dejo el diagrama de conexión eléctrico para estos últimos dos programas si conocen algún programa para hacerlos mas bonitos díganme por que no me gusto mucho como quedo si no pues los haré a mano y los escaneo. Vean los videos de youtube el de como programar con Avr Studio y el de este tercer programa funcionando aquí están los links. O visiten www.mecatronciamexicana.blogspot.com Programar http://www.youtube.com/watch?v=p58QXXSYgXc Tercer Programa http://www.youtube.com/watch?v=FaVF6jDs74o 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 www.mecatronicamexicana.blogspot.com si les gusto o no el tutorial. Muchas gracias por tomarse el tiempo para leer esto suerte. LUIS ANTONIO MÉNDEZ SOTO www.mecatronicamexicana.blogspot.com "AVR®" and "AVR is a registered trademark of Atmel Corporation". un comentario en