sábado, 12 de noviembre de 2016

TrumScript: El lenguaje de programación de Donald Trump

Sam Shadwell y Chris Brown participaron en un Hackaton en la Universidad de Rice, bajo el lema "Make Python Great Again", en 20 horas desarrollaron un nuevo lenguaje de programación: TrumpScript. En ese entonces (principio de año) el país ya se estaba dividiendo entre pro y anti Trumps, y estos programadores quisieron dar su mensaje, inventando este lenguaje.

En la página de Sam se describe la misión de TrumpScript:

"TrumpScript es un lenguaje basado en el ilustre Donald Trump..", "... descubrimos que en el campo actual de los lenguajes de programación, no hay ninguno que aprobaría el dorado y glorioso combover de Trump".

Este es un lenguaje que el mismo Donald Trump aprobaría. Debido a que Donald Trump quiere hacer América grande otra vez, este lenguaje pretende hacer a la programación grande otra vez.

Algunas de las características (y de las más divertidas) son las siguientes:

  • No existe punto flotante, solo enteros, América nunca deja las cosas a media hacer.
  • Todos lo números debe ser estrictamente mayores a 1 millón. Las cosas pequeñas son insignificantes para los americanos. 
  • No se permite ningún tipo de importación. Todo el código tiene que ser casero y americano. 
  • En vez de True o False, se utiliza Fact y Lie (Hecho y Mentira).
  • Se pueden utilizar  solo se pueden utilizar como nombre de variables las palabras favoritas de Trump y nombre de políticos actuales.
  • Todo programa debe terminar con America is great.
  • Los mensajes de error son frases hechas por Trump.
  • No es compatible con Windows, ya que Trump no es una clase de tipo que confiaría en una PC. 
  • Es insensible a mayúsculas o minúsculas.
  • Si se corre el programa desde una computadora en China o Mexico, no se compilará. Prefieren que no roben los secretos tecnológicos de América.
  • No funciona en modo root, ya que América no necesita ayuda ya que es grande. Trump es todo lo que ellos necesitan.

 Instalación

Para instalar se debe descargar el zip desde github haciendo:

git clone  https://github.com/samshadwell/TrumpScript.git

Se debe agregar al PATH del sistema haciendo:








Temporalmente: export PATH=$PATH:/path/to/

TrumpScript/bin to terminal.
Permanentemente:  export PATH=$PATH:/path/to/TrumpScript/bin al final de tu ~/.bashrc file.



 Fuente del lenguaje: https://github.com/samshadwell/TrumpScript





martes, 25 de octubre de 2016

Semáforos vs Mutex - FreeRTOS

El semáforo sirve para sincronizar tareas. Es una simple Queue, de extensión 1. Puede estar vacía o llena. La tarea va a tratar de leer el semáforo y lo verá vacio, entonces se bloquea. Cuando llega la interrupción, esta da ('give') el token al semáforo, es decir llena la queue. Entonces ahora la tarea que estaba bloqueada ya puede venir a tomar ('take')​ el token del semáforo. Realiza su tareas. Al finalizar, vuelve a leer el semáforo, al no tener nada se bloquea.

En el semáforo, la tarea no devuelve el token, es decir que no vuelve a rellenar la queue (el semáforo)
Para aquellos que recién comienzan con el desarrollo en freeRTOS, y en particular a aquellos que nos gusta leer la teoría, antes de ir a la programación, se puede dar una pequeña confusión entre Mutex y Semáforos, ya que aparentemente utilizan el mismo medio: Una cola de lenght 1.

La Mutual Exclusion, es una Técnica, que permite que no exista una inconsistencia en la utilización de recursos compartidos. Y se pueden utilizar varias estrategias para asegurar la Mutual Exclusion.
  1. Entrando en secciones críticas. 
  2. Suspendiendo el scheduler. A mi parecer es lo mismo que el anterior. 
  3. Utilizando Mutex.
Mutex es implementado a través de una especie de semáforo (Queue de extensión 1 que puede estar vacia o llena). La diferencia radica en que las tareas que utilizan un recursos compartido, primero van a mirar el mutex, si está "lleno" toman el token (y dejan el semáforo vacio). Si otra tarea quiere tocar el recurso, verá que el mutex está vacio, por lo tanto se bloquea.

Aqui va la diferencia primordial: para el caso del MUTEX, se tiene que devolver el token. Es decir, esta especie de "semáforo", se vuelve a llenar.

lunes, 24 de octubre de 2016

ARGENTOS, Nuevo lenguaje esotérico bien argento

Introducción

Argentos nació como producto del concurso CALESYTA 2016 (Concurso Argentino de Lenguajes Esotéricos y Tarpits). En un principio iba a ser un lenguaje más “clásico”, pero me aburrí de lo mismo.

Así fue que en una tarde, mientras desarrollaba unas de mis partidas de ajedrez más importante… bueno no, iba en el bondi, y en una de esas epifanías que tienen los que hacen programación, me surgió la idea de hacer una tablero de ajedrez, y tratar de programar con el movimiento de una de las piezas. Me pareció interesante hacerlo con el caballo… hacer… el lenguaje.

Características principales

  • El control de todo el programa se realiza moviendo una pieza del ajedrez: el caballo.
  • Existen tres campos de movimiento, basados en el tablero de ajedrez: el principal, el de variables y el de caracteres.
  • Las coordenadas de los campos (8x8) son 'A’, ‘R’, ‘G’, ‘E’, ‘N’, ‘T’, ‘O’, ‘S’.
  • Las variables se almacenan en el campo de variables (matriz 8x8), solo se pueden almacenar 64 variables.
  • Las variables son de 1 Bytes.
  • Para asignar valores a las variables, se debe utilizar el campo de caracteres. 
  • Los únicos flujos de control son: condicional y repetitivo.

 Fundamentos

Para lograr cumplir con lo esperado de un lenguaje, como control de flujo, almacenar variables, realizar operaciones; fue necesario crear varios campos o tableros de ajedrez.

Para ello  ARGENTOS proporciona un campo de movimiento, que llevaría el flujo principal del programa, y donde se encuentran los códigos principales de control (salto de linear, retorno de carro, espacio, +, -, <, >, =, etc).

Para hacerlo más interesante, la libertard del programador tendría ciertas limitaciones, lo que representa un divertido desafío.

En primer lugar todos los datos que se ingresan en un programa (caracteres, numeros) ya son previstos por ARGENTOS. Para almacenar datos en memoria, también tiene sus limitaciones. Los datos se almacenan en un tablero 8x8 y es necesario recordar la posición en la que esa variable fue almacenada.

 El campo donde se encuentran los caracteres se llama campo de caracteres y el campo donde se almacenan las variables se llama campo de variables.

Campo de movimiento

Argentos es un lenguaje basado en el movimiento del caballo del ajedrez. Se debe imaginar un tablero 8x8, cuyas coordenadas son las letras ‘A’, ‘R’, ‘G’, ‘E’, ‘N’, ‘T’, ‘O’, ‘S’. A continuación se muestra la representación:




Las coordenadas de cada casillero se nombra FILA x COLUMNA. Es decir, el primer casillero (el (0,0)) tiene la coordenada [A,A].

El caballo inicia desde la posición normal del caballo izquierdo, en la posición SR.




El caballo se debe mover respetando el movimiento del ajedrez (en L) .

Rellenando el campo de movimiento


Todos los casilleros se rellenan siguiendo el código ASCII.

Desde el casillero [A,A] hasta el casillo [N,T] está formado por los símbolos de control, incluyendo los operadores matemáticos y lógicos. El mapeo se muestra en la siguiente figura.


 

Campo de caracteres


El campo de caracteres es utilizado para cuando se necesita imprimir por pantalla una cadena de caracteres, o bien almacenar en memoria una cadena. También en esta tabla se encuentran los números.

La idea es la misma, un tablero 8x8, el caballo comienza en la coordenada [S,R].



Para ingresar al campo de caracteres se debe llevar al caballo a la posición [T,A].

Primer Programa

ARCHIVO

El archivo fuente escrito en argentos, debe tener la extensión .argentos.


Ejemplo:
holamundo.argentos

 

ARGENTOS

Todo programa debe empezar y terminar con el comando ARGENTOS.

ARGENTOS

% Tu código aquí

ARGENTOS

Una vez colocado el comando inicial, y antes de cualquier sentencia, se debe mover el caballo hacia cualquier posición, es decir abrir el juego. Para finalizar el script, antes del comando final ARGENTOS, se debe volver al caballo a la posición inicial.

Comandos reservados

Todos los comandos se forman con la combinación de los siguientes caracteres: ‘A’, ‘R’, ‘G’, ‘E’, ‘N’, ‘T’, ‘O’, ‘S’. A continuación se listan los comandos reservados:


argentos Comando para iniciar y finalizar el script
argento Comando para mover el caballo. Comando (espacio) coordenada. Ej: argento tg
ss Comando para ingresar a los campos (de variables, caracteres, etc.).
oo Comando para salir de los campos (de variable, caracteres, etc.).
aaa Con este comando se aceptan las sentencias, coordenadas
ge Print. Se empieza y terminar con el comando ge. Ej “ge ‘algo para print’ ge” → algo para print
rr Se indica que se va a almacenar una variable, en la posición actual del puntero.
nn Se indica que se va a leer una variable de posición actual del puntero.
tt Aumenta el puntero del campo de variables
gg Diminuye el punter del campo de variables
arg Asigna valor a variable (=)
ent Extrae variable

Comentarios

Los comentarios empiezan con %.


% Esto es un comentarios

%Estas son dos líneas

% de comentario

Hola Mundo

A continuación se muestra el clásico “Hola Mundo”

ARGENTOS

argento tg

ge

argento ER argento TA aaa %ingreso al campo de variables

argento OG argento NR aaa % 'h'

argento OG argento TA aaa % 'o'

argento NG argento TN argento OO argento NT aaa  % 'l'

argento TE argento EG aaa %'a'

oo% 'salgo del campo de caracteres'

argento NA aaa % 'espacio'

argento TG argento ER argento TA aaa % Ingreso al campo de caracteres

argento OG argento TN argento NO aaa % 'm'

argento TN argento ET argento TO aaa  % 'u'

argento ET argento NS aaa % 'n'

argento ET aaa % 'd'

argento NE argento OG argento TA aaa %'o'

ge %imprimo


ARGENTOS


Salida:

hola mundo


Campo variables


ARGENTOS brinda una pila 64 Bytes para el almacén de variables. Todas las variables se almacenan dentro de esta pila. No se puede definir variables. Para acceder a esas variables es neceario la utilización de un puntero:

rr Se indica que se va a almacenar una variable, en la posición actual del puntero.
nn Se indica que se va a leer una variable de posición actual del puntero.
tt Aumenta el puntero del campo de variables
gg Diminuye el punter del campo de variables


Es importante recordar a qué posición apunta el puntero de variables. AL comenzar el puntero apunta a la posición 0. Se utiliza el comando rr para indicar que se va a almacenar una variable

En el ejemplo siguiente se muestra como almacenar dos variables en el campo de variables:

%…
nn argento OE argento NN argento argento OT aaa %Se almancena el 1 en la posición 0
tt %Se aumenta el puntero del campo de variables
nn argento SE aaa % Se almacena el 7 en la posición 1 del campo de variables
tt

%…

Flujo selectivo


El lenguaje ARGENTOS permite llevar a cabo un control selectivo. Esto es similar al IF en los demás lenguajes clásicos (como C/C++, Python, etc).

Para empezar un flujo selectivo se debe mover la pieza dos veces, seguidos de tres caracteres, operador de la izquierda, operando y el operador de la derecha. Para salir del flujo selectivo se debe volver a la posición de inicio del flujo.

En pseudo-lenguaje sería de la siguiente manera:

Mover una posición de la pieza

Mover una posición de la pieza

Aceptar la secuencia con ‘aaa’

Ingresar al campo de variables o campo de caracteres

Ingresar el primer operador y aceptar con ‘aaa’

Salir del campo y buscar dentro del campo principal el operando que corresponda. Aceptar con ‘aaa’

Ingresar al campo de variables o campo de caracteres

Ingrear el segundo operador y aceptar con ‘aaa’

Aceptar todo el flujo con ‘aaa’

Escribir el código que corresponda

Mover la pieza del caballo al punto inicial del flujo.

Aceptar la secuencia con ‘aaa’

 

 Flujo repetitivo

El flujo repetitivo es similar al flujo selectivo. Para ingresar a este se debe mover tres posiciones de la pieza. Luego ingresar la cantidad de veces a repetir. Para finalizar el flujo se debe volver al punto de inicio del mismo.

En pseudo-lenguaje sería de la siguiente manera:

Mover una posición de la pieza

Mover una posición de la pieza

Mover una posición de la pieza

Aceptar la secuencia con ‘aaa’

Ingresar al campo de variables o campo de caracteres

Llegar hasta el valor deseado (tiene que ser un número necesariamente) aceptar el valor con ‘aaa’

Aceptar todo el flujo con ‘aaa’

Escribir el código que corresponda

Mover la pieza del caballo al punto inicial del flujo y finalizar el flujo con ‘aaa’.


Conclusión 

Hasta ahora está un poco (¿?) inmaduro pero va tomando forma, en estos días estaré actualizando el post, de manera  de ir completandolo. El compilador, ese es el problema.

miércoles, 12 de octubre de 2016

Resolución de sistemas de ecuaciones lineales con Scilab

Supongamos que tenemos un sistema de ecuaciones lineal del tipo:

 

Lo que se hará básicamente resolver la ecucación de la forma 

A * x = b

Dónde A es la matriz, usualmente cuadrada, conformada por los

Primero, definimos una matriz A que contenga los coeficientes. b es una matriz columna conformada por los resultados del sistema. 

Hacemos:

A = [1 1 2; 2 4 -3; 3 3 -1];
b = [1; 2; 0];

Para resolver esto hacemos:

X = linsolve(A, -b)

o

X = A\b

Y obtenemos como resultado:

x = - 1.3571429 
y =   1.5       
z =   0.4285714  

jueves, 6 de octubre de 2016

nRF24L01 para Arduino (AVR), ejemplo con structs

En el último post de esta serie de publicaciones con respecto al módulo de RF nRF24L01, hemos transmitido un simple entero desde un Arduino Nano a un Arduino Mega2560. 

En este caso veremos un pequeño ejemplo, que permitirá transmitir un struct, que estará conformado por 
  • ID del paquete
  • Cantidad de milisegundos en el que fue enviado el paquete.
El paquete saldrá del Arduino Nano, llegará al Mega el cual simplemente lo recibe, lo imprime en consola por medio del protocolo UART y luego lo reenvía. por su lado el Arduino Nano, se queda esperando un determinado lapso de tiempo. Cuando recibe el paquete que envió calcula la diferencia de milisegundos y la imprime en consola (UART).

La estructura del ejemplo es muy parecida a la del post anterior (sería conveniente volverlo a mirar). Desde el lado del transmisor usamos el struct siguiente:

1:  struct Data_struct {  
2:       uint8_t id;  
3:       uint64_t timing;  
4:  } data_to_send, data_responce;  


Se usará data_to_send para el struct que se envía; y se usa data_responce para el mensaje que se recibe. Se podría usar el mismo y optimizar el código pero opté por mantenerlo simple (y no tan eficiente).
Por el lado del receptor el struct es el siguiente:

1:  struct Data_struct {  
2:       uint8_t id;  
3:       uint64_t timing;  
4:  } data_to_read;  

Envío de datos

Para envíar el dato, primero conformamos el struct. asignamos un ID y llamamos  a la función millis():

1:  data_to_send.id = i;  
2:  data_to_send.timing = millis();  
3:  i++;  
4:  /*Go to TX_MODE*/  
5:  TX_MODE(&data_to_send);  
6:  /*Sending data*/  

Espero mientras se termina el proceso de envío y luego paso al modo de recepción:

1:  /*Wait for transmission end*/  
2:  while(nrf24_isSending());  
3:  RX_MODE(); /*Go to RX MODE*/

La variable lastTiming la utilizo para almacenar el instante en el que envío el paquete (Nota: no es lo más eficiente. Solo es una prueba.)

Si recibo el dato, calculo el tiempo que demora y lo imprimo en pantalla.

Si pasa más de 10000 ms imprimo un mensaje de error, de no llegada de paquetes.


1:     lastTiming = millis();  
2:            do{  
3:                 if(rf_data_ready())  
4:                      {  
5:                           rf_get_data(&data_responce);  
6:                           put_string("\nData recieve correctly\n");  
7:                           put_int(data_responce.id);  
8:                           put_string("\n");  
9:                           put_string("El mensaje tardo: ");  
10:                           put_long(millis() - data_responce.timing);  
11:                           break;  
12:                      }  
13:            }while((lastTiming - temp) < 10000);  
14:            if((lastTiming - temp) > 1000){  
15:                 put_string("\nSending Fails\n");  
16:            }  

Recepción de datos

Para la recepión de datos (en el Arduino Mega2560) simplemente se reciben los paquetes, los imprimo en pantalla y luego se los reenvía, sin realizar ninguna modificación.

1:  if(rf_data_ready())  
2:            {  
3:                 rf_get_data(&data_to_read);  
4:                 put_int(data_to_read.id);  
5:                 put_string("\n");  
6:                 put_long(data_to_read.timing);  
7:                 put_string("\n");  
8:                 TX_MODE(&data_to_read);  
9:                 /*Wait for transmission end*/  
10:                 while(nrf24_isSending());  
11:                 RX_MODE();  
12:            }  


Resultados

El resultado de la ejecución se la ve en la siguiente imágen:


El código del receptor se lo puede encontrar aquí.
El código del transmisor se lo puede encontrar aquí.
 

jueves, 29 de septiembre de 2016

Driver nRF24L01 para arduino

En el post anterior vimos como conectar correctamente los pines del nRF24L01 y los Arduinos Nanos y Mega2560. La conexión de los pines pueden variar dependiendo de la configuración del driver. Pero la que usé en mi ejemplo, es casi como un "estándar", ya que ví varios ejemplos de otros desarrolladores, que lo configuraban como yo.

Vamos a ver las partes más importantes del driver.

En primer lugar en el archivo nRF24L01_RegisterMap.h se encuentra la declaración de los registros del módulo.


Lo primero que se debe hacer, en el main es llamar a la función rf_init() para inicializar el módulo.

1:  //inizialize rf tx   
2:  rf_init();  

Está función hace lo siguiente:

Lo que hace es setear como salida los pines CE y CSN definidos en nRF24L01_RegisterMap.h. Luego pone en cero CE y en uno CSN

Paso siguiente se llama a la función rf_config() en el cual se pasa como parámetro el canal de transmisión más el ancho del payload. Esta función realiza una serie de configuraciones necesarias y básicas para poder funcionar. Y luego se setea en modo RX, para que el módulo se ponga a escuchar.


Es importante setear correctamente las direcciones. Para el transmisor se utiliza las sigiuentes direcciones:

1:  uint8_t tx_address[5] = {0xE7,0xE7,0xE7,0xE7,0xE7};/**<Dirección del transmisor*/  
2:  uint8_t rx_address[5] = {0xD7,0xD7,0xD7,0xD7,0xD7};/**<Dirección del receptorr*/  

En el caso del receptor las direcciones son las siguientes:

1:  /*define addr*/  
2:  uint8_t tx_address[5] = {0xD7,0xD7,0xD7,0xD7,0xD7};  
3:  uint8_t rx_address[5] = {0xE7,0xE7,0xE7,0xE7,0xE7};  

Para ello se utiliza las siguientes función:

1:  //set the device addr  
2:  set_rx_address(rx_address);  
3:  set_tx_address(tx_address);  

Notese que las direcciones están cruzadas.

Transmisor

El código del transmisor se muestra a continuación.


Receptor

El código del receptor se muestra quedó de la siguiente manera.



En post posteriores se moestrará más detalle del código del módulo. Y se buscará complejizar el código.

El código del transmisor se lo puede descargar de aquí. Mientras que al código del receptor se lo descarga desde aquí.

miércoles, 28 de septiembre de 2016

Lenguajes de programación esotéricos

Los lenguajes de programación esotéricos, son lenguajes minimalistas, de naturaleza ofuscada u otra característica que no es común en otros lenguajes de programación. Se suelen usar como prueba de concepto en la creación de lenguajes de programación Turing completos. Poseen una sintaxis muy básica. Estos lenguajes suelen ser creados por simple diversión, ya que difícilmente sean  implementados en proyectos. Sin embargo, se trata de un lenguaje que funcionan como desafío tanto para quienes lo crean, como para aquellos que intentan programar con ellos (Wikipedia).

Existe una wiki, llamada esolangs dedicada a la promoción y documentación de estos lenguajes raros.

Algunos lenguajes esotéricos

Brainf*ck

Este ese el más conocido de estos lenguajes. Fue diseñado por Urban Müller en 1993, con el objetivo de crear un lenguaje simple, turing completo y que requiera de un compilador pequeño.

Aquí se puede ver un ejemplo del famoso "Hola Mundo"

Ook!

Está basado en brainf*ck, pero utiliza en su lugar la combinación de palabras Ook?, Ook! y Ook. Es una parodia del bibliotecario de Mundodisco, que es transformado en orangután y solo puede pronunciar esas tres palabras.

A continuación se observa el ejemplo del "Hola Mundo"


Qriollo

Qriollo es un lenguaje de programación esotérico argentino. El compilador de Qriollo está desarrollado en Haskell, tiene múltiples backends, y puede generar código C, código Python, y bytecode para la JVM. La implementación se encuentra en fase inestable/experimental y cuenta con una interfaz de funciones gringas.

Tiene las funcionalidades usuales de un lenguaje de programación de la familia de ML: tipos de datos algebraicos, análisis de casos por comparación de patrones, clausuras léxicas, optimización de invocaciones a la cola, referencias mutables y call/cc.



martes, 27 de septiembre de 2016

nRF24L01 usando arduino (AVR) II

En el post anterior di una muy breve introducción al módulo nRF24L01. En esta oportunidad daré un poco más de detalle sobre cómo conectar el módulo a los Arduinos.

Antes de continuar se debe indicar que este será usado en el desarrollo de un Drone (quadcopter) casero. El control remoto (Arduino Nano - atmega328p) y el drone en sí mismo (Arduino Mega - atmega2560) se comunicarán por radio frecuencia (RF), usando este módulo.

 

 Pines del nRF24L01+

El módulo tiene 8 pines. 
  1. GND
  2. VCC
  3. CE
  4. CSN
  5. CSK
  6. MOSI
  7. MISO
  8. IRQ

Pines nRF24L01+


VCC siempre tiene que conectarse a 3.3V.
 
La conexión al Arduino Nano es la siguiente:
GND   ->  GND
VCC    ->  3.3V
CE      ->  D9
CSN   ->  D10
SCK   ->  D13
MOSI ->  D11
MISO ->  D12
IRQ    ->  No usado


Arduino Nano + nRF24L01



 Por otro lado tenemos el Arduino Mega conectado de la siguiente manera:

GND   ->  GND
VCC    ->  3.3V
CE      ->  9
CSN   ->  53
SCK   ->  52
MOSI ->  51
MISO ->  50
IRQ    ->  No usado

Arduino Mega + nRF24L01

El código del Arduino Nano (transmisor) se lo puede descargar desde aquí.

El código del Arduino Mega (receptor) se lo puede descargar desde aquí.

En el siguiente post se explicará las principales partes del código.

lunes, 26 de septiembre de 2016

nRF24L01 usando arduino (AVR) I

El nRF24L01 es un tranceiver de 2.4Ghz con un protocolo embebido (Enhanced ShockBurst™). Este se suele utilizar en aplicaciones wireless de bajo consumo. En este caso será utilizado en la comunicación con un Drone.


Resultado de imagen para nrf24l01



Para controlar el chip se utiliza SPI (Serial Peripheral Interface). Los registros son accedidos mediante SPI.

El radio usa modulación GFSK. Este tranceptor tiene parámetros configurables como el canal de frecuencia, power de salida, air data rate (250kbps, 1Mbps, 2Mbps).

La idea de este pequeño proyecto es enviar datos desde un arduino Nano a un arduino Mega, mediante este tranceptor y utilizando el lenguaje C.

Se puede encontrar más información en el datasheet del módulo

domingo, 25 de septiembre de 2016

Blink Led Arduino usando C

En mi primer Post en este Blog, voy a mostrar un proyecto simple. El "Hello Word" de Arduino, pero escrito en C.

Tenemos un entorno en eclipse más el toolchain de AVR.

El código es bastante simple.

En primer lugar definimos las bibliotecas que vamos a utilizar:

#include <avr/io.h>
#include <util/delay.h>


Definimos un MACRO que define el F_CPU, la cual es una manera de indicar cuán rápido está corriendo el MCU.

#define F_CPU 16000000UL

Siguiente paso definimos una función denominada configHardware(), la cual define el pin 13 del arduino Mega como salida:

DDRB |= (1<<PORTB7)

Por último se define el main. Se coloca todo en while infinito. Y se utiliza

PORTB |= (1<<PORTB7) ;

Para prender el led. Y:

PORTB &= ~(1<<PORTB7);


Para apagarlo.


La siguiente lína es un delay de 100 microsegundos.

_delay_ms(100);


Una vez terminado el código queda como sigue:


Compilamos y programamos el arduino y listo.


Dejo el repositorio del ejemplo para decargarlo:

Ejemplo BlinkLed en GitHub