entrega 2

TETRIS HACIENDO USO DEL NUCLEO STM32

Abstract: The following text describes the operation of the first phase of the code for a game of Tetris using the STM32411RE module is detailed as a processing module. For the visualization of the game, an 8X8 LED matrix connected to a MAX7219 was used as a multiplexing system under the language of C ++ programming.

Resumen: A continuacion se detalla el funcionamiento de la primera fase del codigo para un juego de tetris usando el modulo STM32411RE, como modulo de procesamiento, para la visulaizaion del juego se uso una matriz LED 8X8 conectada a un MAX7219 como sistema de multiplexacion bajo el lenguaje de programacion C++.

Keywords: Sistemas Embebidos, STM32, C++, Tetris, Matriz LED, MAX7219

/media/uploads/djinn77/captura.png

SMT32411 RE https://os.mbed.com/platforms/ST-Nucleo-F411RE/

/media/uploads/djinn77/max719.png

MAX7219 y matrix LED

https://www.sparkfun.com/datasheets/Components/General/COM-09622-MAX7219-MAX7221.pdf

Descripcion del protocolo SPI

El Protocolo SPI (Serial Peripherical Interface) o en español Interfaz Periferica Serial es un protocolo síncrono que trabaja de modo full dúplex para recibir y transmitir información, permitiendo que los dos dispositivos pueden comunicarse entre sí al mismo tiempo utilizando canales diferentes o líneas diferentes en el mismo cable. Al ser un protocolo síncrono el sistema cuenta con una línea adicional a la de datos encargada de llevar el proceso de sincronismo.

Dentro de este protocolo se define un maestro que será aquel dispositivo encargado de transmitir información a sus esclavos. Los esclavos serán aquellos dispositivos que se encarguen de recibir y enviar información al maestro. Existen cuatro líneas lógicas encargadas de realizar todo el proceso:

• MOSI (Master Out Slave In):. Línea utilizada para llevar los bits que provienen del maestro hacia el esclavo. • MISO (Master In Slave Out):. Línea utilizada para llevar los bits que provienen del esclavo hacia el maestro. • CLK (Clock):. Línea proviniente del maestro encarga de enviar la señal de reloj para sincronizar los dispositivos. • SS (Slave Select):. Línea encargada de seleccionar y a su vez, habilitar un esclavo.

Descripcion de la conexion

Para poder usar el codigo primero que todo es necesario realizar las correctas conexciones entre el Nucleo STM32 y la matriz led, debemos aseguranos de conectar el PIN MOSI, el Habilitador (CS), y el clock de la siguiente manera

/media/uploads/djinn77/conexin.png

/media/uploads/djinn77/max.png

Conexion al MAX7219

/media/uploads/djinn77/max2.png

Conexion al nucleo STM32

Descripcion del programa

El programa "Tetris_PC"cuenta con dos partes un programa principal "tetris_pc.cpp", que contiene y hace uso de todas las funciones que se usaran para el funcionamiento del programa y adiconal a este se encuentra un archivo de encabezado "piezas. h" que en el cual se encuentran todas las figuras que se utilizan en el juego, como se muestra acontinuacion:

/media/uploads/djinn77/pc.png

Descripcion del Codigo

Como se mensiono anteriormente el programa cuenta con una cabezera en la cual se encuentran configuradas cada una de las piezas que se mostraran sobre la matriz LED al ser llamadas por el programa principal de maera similar a para una figura tipo L, definida por un vector de tamaño 8, con tres posiciones principales que almacenan la forma de la pieza

Encabezado

Encabezado piezas.h

#ifndef PIEZAS_H  
#define PIEZAS_H  
#include "mbed.h"
 
//FORMA DE LA PIEZA
 uint16_t PZA_LD[8]={0b0010000000,  
                     0b1110000000,
                     0b0000000000,
                     0,0,0,0,0};

#endif // PIEZAS_H

Nota: Cabe destacar que para el funcionamiento de la matriz led y por facilidad para el majeno de bits la figura siempre se desplazara de izquierda a derecha.

Programa Principal

Esta seccion encontraremos la descripcion del programa principal, este hace uso de un gran numero de funciones para generar la lectura de sus datos y su posterior procesamiento.

Al principio del programa encontraremos todo lo referente al llamdo de las librerias necesarias para el funcionamiento del programa adicional a ello encontraremos , el establecimiento de las conexiones seriales y la definicion de los pines para cada uno parametros del parametros del protocolo SPI.

Declaracion de Funciones

#include "mbed.h"
#include "piezas.h"
//#define DEBUG 1
 
//Establecimiento de la comunicacion
Serial pc(USBTX,USBRX);            //Establecimiento de la comunicacion serial
SPI deviceM(PB_15, PB_14, PB_13);  //Se defienen los datos que realizaran la comunicacion serial
DigitalOut ssel (PB_12);


Como lo indica su nombre la funcion principal es la mas importante del programa desde ella se iniciara la ejecucion de los demas programas, para esto hace un primer llamamiento a la fucnion de inicializar(), que al ejecutarce establecera como se comporatara el MAX7219, como el modo de trabajo y su luminosidad, posterior ha eso dentro de un while infiniro hace el llamado a la funcion de captura de datos queejecutara la captura de datos para su posterior ejecucion.

Funcion Principal

 
int main() {
    inicializar(); // Inicializa el programa para establecer los modos de trabajo liminosidad
    while(1){
            captura_datos(); // Inicia la lectura de la informacion
            enableizq=1;     // reactiva los desplazamientos
            enableder=1;     // reactiva los desplazamientos
            perder();             // detecta cuando se ha perdido el juego
            }
}


En base a lo anterior el primer llamado de la funcion principal es a esta funcion inicializar(), que establece el modo de trabajo del MAX 7219 como se detalla acontinuacion, posterior ha eso realiza un limpieza de pantalla que evita que se mantengan led encendidos luego de reinicializar el programa.

Funcion inicializar


void inicializar(){ //INICIALIZA LA MATRIZ 
    
    sendSPI(0x0c,1);    
    sendSPI(0x0b,7);
    sendSPI(0x09,0);    //SELECCIONA EL MODO DE TRABAJO DE LA MATRIZ
    sendSPI(0x0A,0x0f); //SELECCIONA LA LUMINOSIDAD DE LA MATRIZ
    int i;
    for (i=0;i<2;i++){
        sendSPI(0x0F,1);
        wait (0.5);
        sendSPI(0x0f,0);
        wait (0.5);
    }
    
    for (int j= 1; j<=8;j++){  // limpia la pantalla al encenderce o reiniarcea asi 
                               //no quedan leds encendidos cuando se ejecute el programa nuevamente
          
          sendSPI(j, 0x00);    //pone cada columna y vecto en blanco al inicializar
                            } 
    }

La funcion de lectura es la más laga de todas desde aqui se capturan los datos para ser procesado, estableciendo una comunicacion serial de 38400 bauds, enviandolos a las variables de posicion (pos), figura y rotacion.

Debe Ingresar 3 datos al abrir el send string del coolterm el primero indica la posicion , luego la figura y finalmente el giro en el siguiente formato 00 00 00 en hexadecimal a 38400 bauds, estos comandos se deben ingresar dentro de el equivalente hexadecimal de '<' y '>', es decir 3C y 3E.

/media/uploads/djinn77/comados.png

captura de datos

 
void captura_datos(){
         
         pc.baud(38400); //Inicializa la velocidad de comunicacion
         // Establece las variables que tomaran los comandos
         char inicio=0,final=0;
         figura=0;
         giro=0;
         posicion=0;
         
         debuging("\n Ingrese el inicio del comando. ");
         inicio=pc.getc(); 
         debuging("\n Ingrese la Figura. ");
         figura=pc.getc();
         debuging("\n Seleccione el giro. ");
         giro=pc.getc();
         debuging("\n Seleccione la posicion. ");
         posicion=pc.getc(); 
         debuging("\n Ingrese el final del comando. ");
         final=pc.getc();    
         
         // Detecta si se ha generado algun error al ingresar el comando
         if(inicio!= '<' || final != '>'){
             
             debuging("\n Error en el comando.");
             
             }else{
                 
                 read();
                  
                 }
                 
         
 
    }

Inicializacion del Programa.

Luego de conectar al PC el Dispositivo debemos ir al icono de escritorio y abrir el programa.

/media/uploads/djinn77/coolterm.png

Luego se debe esteblecer la velocidad de conexion del puerto. Esta debe estar en 38400 bauds y en el puerto detectado por el PC con el nucleo STM32.

luego debe hacer clic en conectar.

/media/uploads/djinn77/conectar_q7axQGC.png

Se debe ir a la siguiente opcion.

/media/uploads/djinn77/sendstring.png

Aparecera la siguiente pantalla en la cual se ingresa el comando este debe estar en el siguiente formato

comando = "3C 01 01 01 3E".

/media/uploads/djinn77/comados.png

/media/uploads/djinn77/enviocomando.png

los valores 3C y 3E, indican el inicio y finalizacion del comando, cada uno de estos caracteres es equivalente a '<' y a '>', respectivamente, solo si estos son enviados al inicio y el final del programa seran capturados y ejecutados de manera correcta de lo contraio se preguntara por ellos nuevamente hasta que se envie el comando correctamente

El valor # 2 del comando selecciona una posicion entre 00-07, en la cual se desplegara la figura inicialmente.

/media/uploads/djinn77/pos_C0G1ii1.png

El siguiente comando ingresa la figura tal que: :

00. L,

/media/uploads/djinn77/l.png

01. T,

/media/uploads/djinn77/t.png

02. Linea.

/media/uploads/djinn77/linea.png

03. Cuadrado.

/media/uploads/djinn77/cuadrado.png

04. Z.

/media/uploads/djinn77/zinv.png

Los giros van el siguiente recuadro lo explica para la letra L:

00 = 0° grados.

/media/uploads/djinn77/l.png

01 =90° grados,

/media/uploads/djinn77/ln.png

02 =180° grados,

/media/uploads/djinn77/lo.png

03 = 270°grados,

/media/uploads/djinn77/ldd.png

En su segunda parte como se puede ver en el ejemplo el procesamiento de la variables anteriores con dos ciclos swicth anidados, es realizado, el primero escoje la figura y el segundo la rotacion, una vez dentro de esto swicth se realiza el llamado a la funcion desplazamiento tomando como parametros de entrada al vector de la figura extraido del encabezado y a la posicion en el que este se mostrara en la matriz

Funcion de lectura

void read(){
         
         switch(figura){ //Este switch escoje la figura con que trabajaar y actualiza los limite laterales e inferiores por figura
             case 0: // L
             if(giro == 0)
             imprimir =  PZA_L,liminf=7, limizq=6, limder=0;
             if(giro == 1)
             imprimir =  PZA_LDN,liminf=7, limizq=5, limder=0;
             if(giro == 2)
             imprimir =  PZA_LDO,liminf=7, limizq=5, limder=-1;
             if(giro == 3)
             imprimir =  PZA_LDD,liminf=8, limizq=5, limder=0;
             break;
             case 1://T       
             if(giro == 0)
             imprimir =  PZA_T,liminf=8, limizq=5, limder=0;
             if(giro == 1)
             imprimir =  PZA_TN,liminf=7, limizq=6, limder=0;
             if(giro == 2)
             imprimir =  PZA_TO,liminf=7, limizq=5, limder=0;
             if(giro == 3)
             imprimir =  PZA_TD,liminf=7, limizq=5, limder=-1;
             break;
             case 2://I
             if(giro == 0 || giro == 2)
             imprimir =  PZA_I,liminf=7, limizq=6, limder=-1;
             if(giro == 1 || giro == 3)
             imprimir =  PZA_IR,liminf=8, limizq=5, limder=0;
             break;
             case 3://Cuadrado
             if(giro == 0 || giro == 1 || giro == 2 || giro == 3)
             imprimir =  PZA_C,liminf=8, limizq=6, limder=0;
             break;
             case 4:
             if(giro == 0 || giro == 2)
             imprimir =  PZA_Z,liminf=8, limizq=5, limder=0;
             if(giro == 1 || giro == 3)
             imprimir =  PZA_ZN,liminf=7, limizq=6, limder=0;
             break;
             default:
             imprimir = VACIO;
             break;
             }
            // pc.printf("\n limite inferior read %d\n ",liminf);
             desplazar();
     }    
  
     

Una vez capturados los datos anteriores la funcion de desplazamiento posee un vector del mismo tamaño de la pieza capturada y es inicializado en ceros cada vez que es llamado, la figura es reubicada en base a la posicion deseada mediante el otro dato capturado por la funcion aqui la variable i inicializa desde que i = posicion ingresada hasta que cubre las tres posiciones principales que sobre la figura para luego de organizada ser llevada a la matriz de captura.

Funcion de desplazamiento

void desplazar(){
      
      uint16_t desplazamiento[8]={0};      // Inicia un vector auxiliar con solo Ceros
      int j= 0;
       for(int i=posicion; i<(posicion+3);i++){  // Inicia un vector auxiliar con solo Ceros
           desplazamiento[i]=imprimir[j]; // Alamcena los nuevos datos tomados en el vector 
           j++;                    
                          }
       imprimir=desplazamiento;          //  Actualiza el puntero imprimir con la posicion seleccionada para la fig
       captura_matriz();
      }  
  
  

La matriz de captura es la encargada de memorizar y de realizar la impresion de los datos que ejecutara el programa. cuenta con 2 ciclos while que permiten que se ejecute el codigo haste que cierto parametro no se cumpla, el pimer ciclo while, realizar el corrimiento de los bits del vector capturado hasta la parte inferior de la matriz para luego ser capturados en la memoria, el segundo ciclo while imprime a cada "columna" el corrimento del vector, e l programa cuenta con un sistema que le permite identifcar la colicion con otra figura previamente enviada haciendo uso de la operacion and, mientras recorre el vector el resultado de esta siempre sera 0 hasta que finalemente vale 1 al chocar con otro objeto y luego ser nuevamente almacenado en la memoria todos los valores del la posicion del corrimiento en la memoria.

Funcion de captura de matriz

while(j<=8){ //se encarga de seleccionar que posicion del vector imprimira
         
         sendSPI(j, memoria[j-1]|(imprimir[j-1]>>i));  //Imprime el resultado de aplicar OR a la memoria y al desplamamiento de imprimir
         //wait(0.1); // Activar este wait para pruebas de desplazamiento y captura de los datos
 
     if(i==liminf || (memoria[j-1]&(imprimir[j-1]>>i+1))!=0){  //Detiene el desplazamiento cuando los bits tocan fondo, 
         enable=0;// desabilitador del ciclo while 
         
         for(int k=0; k<8;k++){                          // Almacena todos lo datos del vector en la posicion que corrimiento en la memoria
              memoria[k]= memoria[k]|(imprimir[k]>>i);
             }
    
         }
     j++;
     } 
     wait(velocidad); // espera para tomar el otro valor de lectura
     i++;
     }

Para poder desplazar la figura sobre la matriz es necesario capturar el puntero imprimir el cual almacena temporalmente la figura que se desplaza sobre el mapa, para ello se usa una interrupcion que llama a la funcion en el flanco de bajada y provoca el desplazamiento del vector y la limpieza de los otros valores desplazados

desplazamiento izquierda

void desplazar_izq(){                                       // Desplaza el vector de impresion hacia el lado izq de la pantalla
      if(posicion<limizq && corrimiento<liminf && enableizq){ // Establece los limite de movimiento del vector
          posicion++;                                         // Aumenta el valor de la posicion segun las veces se haya pulsador el boton
            for(int i=7; i >=0; i--){
                imprimir[i]=imprimir[i-1];
                if(i==0){
                    imprimir[i]=0;
                    
                    } 
                if((imprimir[i]&& memoria[i+1])!=0){          // Si esta condicion se cumple desactiva el desplzamiento izq   
                    enableizq=0;
                    }      
                            
                }
            for(int j=1;j<=8;j++){
             sendSPI(j, memoria[j-1]|(imprimir[j-1]>>corrimiento));
 
                 }   
            wait(velocidad);  
          
          }
      return;
      }
  

Similar al anterior ambos codigo desplazan la figura hacia uno de los dos lados y terminan su desplazamiento cuando alguna de las condiciones iniciales se incumplen

desplazamiento derecha

void desplazar_der(){                                           // Desplaza el vector de impresion hacia el lado derecho de la pantalla
      if(posicion>limder && corrimiento<liminf && enableder){     // Establece los limite de movimiento del vector
           posicion--;                                            // Disminuye el valor de la posicion segun las veces se haya pulsador el boton
            for(int i=0; i <8; i++){
 
                imprimir[i]=imprimir[i+1];
                if(i==7){
                    imprimir[i]=0;
                    
                    }
                if((imprimir[i]&& memoria[i-1])!=0){              // Si esta condicion se cumple desactiva el desplzamiento der  
                    enableder=0;
                    }     
                }    
            for(int j=1;j<=8;j++){
                 sendSPI(j, memoria[j-1]|(imprimir[j-1]>>corrimiento));
                 }   
            wait(velocidad); 
      }
      return;
      } 
      
  

el codigo de giro de la figura esta dividido en 2 parte la primera parte es identica a la seleccion inicial de la figura hecho por el cual no se mostrara acontinuacion, sin embargo esta nueva funcion toma utiliza una funcion de retorno para girar la figura y imprimirla en su ultima condicion conocida.

giro de la figura

uint16_t* girar(uint16_t* pieza, int posicion){          // gira la figura y la desplza a ultima posicion conocida
     uint16_t desplazamiento[8]={0};                      
     int j= 0;
     for(int i=posicion; i<(posicion+3);i++){             // Inicia un vector auxiliar con solo Ceros
     desplazamiento[i]=pieza[j];                          // Alamcena los nuevos datos tomados en el vector 
     j++;                    
     }
    
    return desplazamiento;                               // regresa el vector procesado y lo asigna al puntero imprimir
    }
    

Acontinuacion se muestra el codigo en conjunto del desplazamiento, giro e impresion de las figuras.

codigo en conjunto

oid captura_matriz(){
    
int i=0; // inicia el contador i en 0
int enable=1;
 
pc.printf("\n limite inferior captura_matriz %d\n ",liminf);
 while(enable){ //se encarga de desplazar los bits dentro del vector
    //pc.printf("\n1- conteo del corriemiento:\n %d",i);
    corrimiento=i;
    int j=1;  // inicia el contador j en 1        
    pulsador_izq.fall(&desplazar_izq);           // interupcion al activarce el boton izquierdo
    pulsador_der.fall(&desplazar_der);           // interupcion al activarce el boron  derecho
    
    //////////////////////////////////////////777
   while(p_giro == 1){                           // se activa al pulsar boton giro
           giro++;                               // actualiza el valor del giro para seleccionar la figura girada
          
           if(giro ==4){                         // Si giro == 4 reseta el giro para volver a la fig inicial
               giro=0;
               }
           switch(figura){
             case 0: // L
             if(giro == 0)
             imprimir= girar(PZA_L, posicion),liminf=7, limizq=6, limder=0;
             if(giro == 1)
             imprimir= girar(PZA_LDN, posicion),liminf=7, limizq=5, limder=0;
             if(giro == 2)
             imprimir= girar(PZA_LDO, posicion),liminf=7, limizq=5, limder=-1;
             if(giro == 3)
             imprimir=girar(PZA_LDD, posicion),liminf=8, limizq=5, limder=0;
             break;
             case 1://T       
             if(giro == 0)
             imprimir= girar(PZA_T, posicion),liminf=8, limizq=5, limder=0;
             if(giro == 1)
             imprimir= girar(PZA_TN, posicion),liminf=7, limizq=6, limder=0;
             if(giro == 2)
             imprimir= girar(PZA_TO, posicion),liminf=7, limizq=5, limder=0;
             if(giro == 3)
             imprimir= girar(PZA_TD, posicion),liminf=7, limizq=5, limder=-1;
             break;
             case 2://I
             if(giro == 0 || giro == 2)
             imprimir= girar(PZA_I, posicion),liminf=7, limizq=6, limder=-1;
             if(giro == 1 || giro == 3)
             imprimir=girar(PZA_IR, posicion),liminf=8, limizq=5, limder=0;
             break;
             case 3://Cuadrado
             if(giro == 0 || giro == 1 || giro == 2 || giro == 3)
             imprimir= girar(PZA_C, posicion),liminf=8, limizq=6, limder=0;
             break;   
             case 4: //Z
             if(giro == 0 || giro == 2)
             imprimir= girar(PZA_Z, posicion),liminf=8, limizq=5, limder=0;
             if(giro == 1 || giro == 3)
             imprimir= girar(PZA_ZN, posicion),liminf=7, limizq=6, limder=0;
             break;
 
             }
             wait(velocidad);  
             }  
             
    ///////////////////////////////////////////    
    while(j<=8){ //se encarga de seleccionar que posicion del vector imprimira
         
         sendSPI(j, memoria[j-1]|(imprimir[j-1]>>i));  //Imprime el resultado de aplicar OR a la memoria y al desplamamiento de imprimir
         //wait(0.1); // Activar este wait para pruebas de desplazamiento y captura de los datos
 
     if(i==liminf || (memoria[j-1]&(imprimir[j-1]>>i+1))!=0){  //Detiene el desplazamiento cuando los bits tocan fondo, 
         enable=0;// desabilitador del ciclo while 
         
         for(int k=0; k<8;k++){                          // Almacena todos lo datos del vector en la posicion que corrimiento en la memoria
              memoria[k]= memoria[k]|(imprimir[k]>>i);
             }
    
         }
     j++;
     } 
     wait(velocidad); // espera para tomar el otro valor de lectura
     i++;
     }
 
  }
  
  void desplazar_izq(){                                       // Desplaza el vector de impresion hacia el lado izq de la pantalla
      if(posicion<limizq && corrimiento<liminf && enableizq){ // Establece los limite de movimiento del vector
          posicion++;                                         // Aumenta el valor de la posicion segun las veces se haya pulsador el boton
            for(int i=7; i >=0; i--){
                imprimir[i]=imprimir[i-1];
                if(i==0){
                    imprimir[i]=0;
                    
                    } 
                if((imprimir[i]&& memoria[i+1])!=0){          // Si esta condicion se cumple desactiva el desplzamiento izq   
                    enableizq=0;
                    }      
                            
                }
            for(int j=1;j<=8;j++){
             sendSPI(j, memoria[j-1]|(imprimir[j-1]>>corrimiento));
 
                 }   
            wait(velocidad);  
          
          }
      return;
      }
      
  void desplazar_der(){                                           // Desplaza el vector de impresion hacia el lado derecho de la pantalla
      if(posicion>limder && corrimiento<liminf && enableder){     // Establece los limite de movimiento del vector
           posicion--;                                            // Disminuye el valor de la posicion segun las veces se haya pulsador el boton
            for(int i=0; i <8; i++){
 
                imprimir[i]=imprimir[i+1];
                if(i==7){
                    imprimir[i]=0;
                    
                    }
                if((imprimir[i]&& memoria[i-1])!=0){              // Si esta condicion se cumple desactiva el desplzamiento der  
                    enableder=0;
                    }     
                }    
            for(int j=1;j<=8;j++){
                 sendSPI(j, memoria[j-1]|(imprimir[j-1]>>corrimiento));
                 }   
            wait(velocidad); 
      }
      return;
      } 
      
  uint16_t* girar(uint16_t* pieza, int posicion){          // gira la figura y la desplza a ultima posicion conocida
     uint16_t desplazamiento[8]={0};                      
     int j= 0;
     for(int i=posicion; i<(posicion+3);i++){             // Inicia un vector auxiliar con solo Ceros
     desplazamiento[i]=pieza[j];                          // Alamcena los nuevos datos tomados en el vector 
     j++;                    
     }
    
    return desplazamiento;                               // regresa el vector procesado y lo asigna al puntero imprimir
    }
    

Para finalizar el codigo se identifica cuando la memoria ha alcanzado el limite superior denotando una derrota del juego para esto se usa el puntero borde que al tener una equivalencia superior a 128 finaliza y reinia el juego.

condiciones de derrota

void perder(){
     for(int i=0; i<8;i++){                             // detecta si alguna parte de una figura ha tocado el borde superior para reinicializar el juego
        borde= &memoria[i];
        if(*borde>=128){
          NVIC_SystemReset();
          }
        }
    }  

Para mas informacion puede visitar los siguientes enlaces:

1. https://drive.google.com/open?id=18LxjhfQ7sTEd2r316q1CN2qZuNM1WJ5o

2. https://drive.google.com/open?id=1Z5J7cLjWl7QM84lvCdH_pyFUXef9isOn

3. https://drive.google.com/open?id=1fTFKVzpz2CauPSMowCayQmsoqZJYUY8f

4. https://drive.google.com/open?id=1_HB4bgzSwmUMdbu0HE6zHlmdiYrzXcwq

Nota: cada vez que se requiera enviar una figura debe enviarce un nuevo comando desde el coolterm


Please log in to post comments.