Gestión de Puertos

Descargar como docx, pdf o txt
Descargar como docx, pdf o txt
Está en la página 1de 18

GESTIN DE PUERTOS Cada puerto tiene asociado 3 registros: REGISTRO TRIS: Configura las lneas del puerto como

entrada o salida. Con 1, se define como entrada y con 0 como salida. REGISTRO PORT: Permite leer o escribir datos en el puerto. REGISTRO LAT: Es el latch de salida del puerto. En lenguaje C se pueden gestionar los puertos de dos formas: 1. Se declaran los registros TRISX y PORTX definiendo la posicin de la memoria RAM como variable de C. 2. Usando la directivas especficas del compilador (#USE FAST_IO, #USE FIXED_IO, #USE STANDARD_IO). USANDO LA MEMORIA RAM Las posiciones de memoria que utilizan los registros TRISX y PORTX del PIC18 se definen as: #BYTE tris_a = 0xf92 #BYTE tris_b = 0xf93 #BYTE tris_c = 0xf94 #BYTE tris_d = 0xf95 #BYTE tris_e = 0xf96 #BYTE port_a = 0xf80 #BYTE port_b = 0xf81 #BYTE port_c = 0xf82 #BYTE port_d = 0xf83 #BYTE port_e= 0xf84 Definidas las variables se pueden controlar los puertos por comandos de asignacin. Ejemplos: tris_a = 0xff ; //Todo el puerto a como entrada tris_b = 0x00; //Todo el puerto b como salida tris_d = 0x0f; //Los 4 primeros bits del puerto d como entrada y los 4 bits (MSB) como salida. Luego se escribe o lee los datos en el puerto: Escritura del puerto: port_b = 0x0B; Lectura del puerto: // el dato 1011, se coloca en el puerto b

Contador = port_a

// el dato del puerto a se asigna a la variable Contador

Proyecto 1: Prender y apagar los LEDs conectados al puerto B indefinidamente. //MANEJO DE TODO EL PUERTO #include <18F4550.h> #fuses HS,NOWDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #byte port_b = 0xf81 #byte tris_b = 0xf93 void main() { disable_interrupts(GLOBAL); tris_b = 0x00; /* Puerto B como salida */ port_b = 0x00; // Asigna a todo el Puerto B 0 do{ port_b = 0xff; //Coloca todo el puerto B en 1 delay_ms( 500 ); //retardo de 500 ms port_b = 0x00; //Coloca todo el puerto B en 0 delay_ms(500); //Retardo de 500 ms } while ( TRUE ); //Bucle infinito } Proyecto 2: Prender y apagar el LED del puerto RB0 indefinidamente. //MANEJO BIT POR BIT #include <18F4550.h> #fuses HS,NOWDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #byte tris_b = 0xf93 #bit b0= 0xf81.0 void main() { disable_interrupts(GLOBAL); tris_b = 0x00; /* Puerto B como salida */ do{ b0= 0; delay_ms( 500 ); b0 =1; delay_ms(500); } while ( TRUE ); } A TRAVS DE DIRECTIVAS El compilador ofrece funciones predefinidas para trabajar con los puertos. Estas son:

output_x (valor); //Por el puerto indicado saca el valor correspondiente a 0 a 255 input_x(); //Se obtiene el valor del puerto set_tris_x(valor); //valor= TRUE habilita o valor = FALSE, habilita o desabilita las resistencias pull up del puerto B get_tris_x(); //Devuelve el valor del registro tris_x x, es el smbolo correspondiente a los puertos A, B, C, D, E. Las funciones asociadas al terminal o pin son: output_low (PIN_n); //Pin a 0 output_high (PIN_n); //Pin a 1 output_bit (PIN_n,valor);//Pin al valor especificado output_toggle (PIN_n); //Complemento el valor del pin output_float (PIN_n); //Pin de entrada, quedando a tensin flotante. Simula salida de drenador abierto, output_state (PIN_n); //Lee el valor del pin input_state(PIN_n); //Lee el valor del PIN sin cambiar el sentido del terminal. input(PIN_n); //Lee el valor del PIN n se refiere al nombre del puerto y del PIN. Ejemplo: output(PIN_B0) #USE STANDARD_IO(PUERTO) Es la directiva por defecto. Las funciones output_x e input_x, modifican el TRIS asegurando que los terminales usados sean de salida o entrada respectivamente. Proyecto 3: //Prende y apaga un led #include <18f4550.h> //archivo de cabecera #fuses HS, NOWDT, NOPROTECT, PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use standard_io(b) //Directiva usado por el compilador por defecto. Es opcional llamarla en el programa //PROGRAMA void main(void) { disable_interrupts(GLOBAL); //todas las interrupciones desactivadas do{ output_low(PIN_B0); //led off delay_ms(500); output_high(PIN_B0); //led on delay_ms(500); } while(TRUE); //bucle infinito }

#USE FAST_IO(PUERTO) Las funciones output_x e input_x, no modifican el TRIS, por lo cual se debe colocar necesariamente la instruccin para definir que los terminales usados sean de salida o entrada respectivamente. Proyecto 4: //Prende y apaga un led #include <18f4550.h> //archivo de cabecera #fuses HS,NOWDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) //PROGRAMA void main(void) { set_tris_b(0x00) disable_interrupts(GLOBAL); //todas las interrupciones desactivadas do{ output_low(PIN_B0); //led off delay_ms(500); output_high(PIN_B0); //led on delay_ms(500); } while(TRUE); //bucle infinito } #USE FIXED_IO(PUERTO_OUTPUTS=pinx) La directiva indica slo los terminales de salida. El compilador se encarga de generar el cdigo para definir los puertos de acuerdo con la informacin, sin tener en cuenta si la operacin es entrada o salida. Proyecto 5: //Prende y apaga un led /* MANEJO DEL PUERTO USANDO DIRECTIVA #USE FIXED */ #include <18F4550.h> #fuses HS, NOWDT, NOPROTECT, PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fixed_io(b_outputs=pin_b0) void main() { disable_interrupts(GLOBAL); while (TRUE) //Otra forma de generar un bucle infinito

{ output_high(pin_b0); delay_ms( 500 ); output_low(pin_b0); delay_ms(500); } } SECUENCIAS CON LEDS El proyecto consiste en activar y desactivar los LEDs que estn conectados al puerto uno a uno empezando desde el puerto RB0 hasta llegar al RB7 en forma indefinida. Para este caso se procedido asi: 1. Inicializar el puerto B con (00000001), para activar el LED que est conectado en RB0. 2. Mostrar este valor durante un tiempo (100 ms). 3. Desplazar el valor del puerto una psosicin a la izquierda (port_b = port_b<<1). 4. Repetir este proceso hasta que llegue el 1 al puerto RB7. 5. En este punto, el desplazamiento se realiza a la derecha (port_b=port_b>>1) hasta que llegue a RB1. 6. Repetir el proceso. La tabla muestra el estado del puerto b, para la secuencia requerida. RB7 0 0 0 0 0 0 0 1 0 0 0 0 0 RB6 0 0 0 0 0 0 1 0 1 0 0 0 0 RB5 0 0 0 0 0 1 0 0 0 1 0 0 0 RB4 0 0 0 0 1 0 0 0 0 0 1 0 0 RB3 0 0 0 1 0 0 0 0 0 0 0 1 0 RB2 0 0 1 0 0 0 0 0 0 0 0 0 1 RB1 0 1 0 0 0 0 0 0 0 0 0 0 0 RB0 1 0 0 0 0 0 0 0 0 0 0 0 0

0 0 0

0 0 0

0 0 0

0 0 0

0 0 0

0 0 0

1 0 0

0 1 0

Proyecto 6: //Secuencia de encendido de leds desde la izquierda #include <18f4550.h> //archivo de cabecera #fuses HS,NOWDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #BYTE port_b= 0xf81 int i; ///PROGRAMA void main(void) { set_tris_b(0x00); //portb como salida(RB0,las demas desactivadas) disable_interrupts(GLOBAL); //todas las interrupciones desactivadas while(TRUE){ port_b=0x01; for (i=1;i<=8;++i) { delay_ms(100); port_b= port_b<<1; } port_b= 128; for (i=1;i<=8;++i) { delay_ms(100); port_b= port_b>>1; } }//bucle infinito }

CONEXIN DE PULSADORES

Todos los ejercicios realizados hasta el momento, los puertos del PIC se han utilizado como salidas. A continuacin, se presentan proyectos para manejar como entradas. Se conectan pulsadores para cambiar los niveles de tensin en los pines del puerto. La figura muestra las conexiones para manejar pulsadores. Con esta conexin los pines del puerto (RDO y RD1), estn directamente en estado lgico de 1. Por consiguiente, el accionamiento del pulsador determina que pasen los puertos indicados a 0.

El ejercicio consiste en prender un led al accionar un pulsador 1 (P1) y apagar el led al accionar el pulsador 2 (P2). Los pulsadores P1 y P2 estn conectados al puerto D en los pines RD0 y RD1 respectivamente. En el programa se definen las variables prender y apagar mediante las directivas #define prender bit_test(port_d,0), #define apagar bit_test(port_d,1), las cuales permiten testar el estado de las entradas. Proyecto 7: //MANEJO DE PULSADORES #fuses HS,WDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #use fast_io(d) #BYTE port_b= 0xf81 #BYTE port_d= 0xf83 #define prender bit_test(port_d,0) #define apagar bit_test(port_d,1)

///PROGRAMA void main(void) { set_tris_b(0x00); //portb como salida set_tris_d(0xff); //portd como entrada disable_interrupts(GLOBAL); //todas las interrupciones desactivadas while(TRUE){ if (prender == 0) { bit_set(port_b,0); } if (apagar==0) { bit_clear(port_b,0); } }//bucle infinito } Otra forma de sensar el estado de los pines es utilizando la instruccin input(PIN_D0) == 0. Para este caso si el pin del puerto D0 (o cualquier otro) que est en alto, si cambia de estado ser detectado por la instruccin indicada. Proyecto 8: #include <18f4550.h> //archivo de cabecera #fuses HS,WDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use standard_io(B) #use standard_io(D) ///PROGRAMA void main(void) { disable_interrupts(GLOBAL); //todas las interrupciones desactivadas output_low(PIN_B0); while(TRUE){ if (input(PIN_D0) == 0) output_high(PIN_B0); if (input(PIN_D1) == 0) output_low(PIN_B0); }//bucle infinito }

Uno de los problemas que presentan los pulsadores mecnicos al momento de su accionamiento es el rebote de los contactos, lo que genera que se produzcan varios transistorios de voltaje (pulsos) que afectan el funcionamiento normal del circuito. Una forma de eliminar este inconveniente es mediante la inclusin de compuertas smitch trigger, o por software, como se presenta a continuacin. Proyecto 9: ///ACTIVA LED CON PULSADOR 1 Y APAGA LED CON PULSADOR 2. INCLUYE ANTIRREBOTE #include <18f4550.h> //archivo de cabecera

#fuses HS,WDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #use fast_io(d) #BYTE port_b= 0xf81 #BYTE port_d= 0xf83 #define prender bit_test(port_d,0) #define apagar bit_test(port_d,1) void ///PROGRAMA void { set_tris_b(0x00); set_tris_d(0xff); disable_interrupts(GLOBAL); antirebote();

main(void)

//portd //todas

las

como interrupciones

entrada desactivadas

while(TRUE){ if (prender == 0) { antirebote(); bit_set(port_b,0); } if (apagar==0) { bit_clear(port_b,0); } }//bucle infinito } void antirebote(void) { while(prender == 0) { } // No realiza nada hasta que el pulsador est inactivo while(apagar == 0) { } // No realiza nada hasta que el pulsador est inactivo

delay_ms(30); return; } Cada vez que se accione un pulsador se averigua si el estado del puerto sigue siendo 0, de ser afirmativo se llama a una subrutina antirrebote, para que de mantenerse en este estado no realice ninguna accin hasta que el pulsador este inactivo. La instruccin usada para el efecto es while(prender == 0) { }. Una vez que el pulsador este inactivo, de la subrutina retornar (por return) al programa a seguir ejecuntando las instrucciones. Una forma forma prctica de averiguar el antirrebote, es mediente el siguiente proyecto, que consiste, en prender y apagar un LED con un mismo pulsador. En este caso al accionar el pulsador se prende el LED (si est apagado), y al volver a accionar el mismo pulsador se apaga el LED. Proyecto 10: ///PRENDER Y APAGAR LED CON UN MISMO PULSADOR SIN ANTIRREBOTE #include <18f4550.h> //archivo de cabecera #fuses HS,WDT,NOPROTECT,PUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #use fast_io(d) #BYTE port_b= 0xf81 #BYTE port_d= 0xf83 #define prender bit_test(port_d,0) int k; //PROGRAMA; void main() { set_tris_b(0x00); set_tris_d(0xff); //portd como entrada. disable_interrupts(GLOBAL); //todas las interrupciones desactivadas port_b = 0; k= 0; while(TRUE){ if (prender == 0 && k==0) { k= 1; bit_set(port_b,0); } if (prender==0 && k==1) { k=0; bit_clear(port_b,0);

} }//bucle infinito } Se ha utilizado una variable k , con la cual permite prender el LED (k=0) y apagar el LED (k=1). Observe tambin que se efecta una operacin lgica and con el estado de la entrada (prender == 0 && k==0).

TECLADO MATRICIAL En los PICs, una de las formas ms comunes y tiles para ingresar datos es el uso de teclados matriciales. La figura presenta el diagrama de conexiones de un teclado matricial (4 x 4).

La lista de instrucciones muestra el programa para ingresar nmeros del 0 al 9. Un decodificador BCD, se encarga de transformar en 7 segmentos para mostrarlos en el display.

//TECLADO PROVISTO DE ANTIRREBOTE


#include <18f4550.h> //archivo de cabecera #fuses HS, WDT, NOPROTECT, NOPUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #use fast_io(d) #BYTE port_b= 0xf81 #BYTE port_d= 0xf83 #BIT FA = 0xf83.0 #BIT FB = 0xf83.1 #BIT FC = 0xf83.2 #BIT FD = 0xf83.3 #define C1 bit_test(port_d,4) #define C2 bit_test(port_d,5) #define C3 bit_test(port_d,6) #define C4 bit_test(port_d,7) void antirebote(); int x; ///PROGRAMA; void main() { set_tris_b(0x00);//Puerto b como salida set_tris_d(0xf0); //Puerto d 4 primeros bits como salida y 4 bits como entrada. disable_interrupts(GLOBAL); //todas las interrupciones desactivadas port_b = 0; FA=1; FB=1; FC=1; FD=1; while(TRUE){//BARRIDO FA=0; if (C1 == 0){ x=1; antirebote(); } if (C2 == 0){ x=2; antirebote(); } if (C3 == 0){ x=3; antirebote(); } if (C4 == 0){ x=10; antirebote(); } FA=1; // FB=0; if (C1 == 0){ x=4; antirebote(); } if (C2 == 0){ x=5; antirebote(); } if (C3 == 0){ x=6; antirebote(); } if (C4 == 0){ x=11; antirebote(); } FB=1; // FC=0; if (C1 == 0){ x=7; antirebote(); }

if (C2 == 0){ x=8; antirebote(); } if (C3 == 0){ x=9; antirebote(); } if (C4 == 0){ x=12; antirebote(); } FC=1; // FD=0; if (C1 == 0){ x=13; antirebote(); } if (C2 == 0){ x=0; antirebote(); } if (C3 == 0){ x=14; antirebote(); } if (C4 == 0){ x=15; antirebote(); } FD=1; }//bucle infinito } void antirebote(void) { while(C1 == 0) { } // No realiza nada hasta que el pulsador est inactivo while(C2 == 0) { } while(C3 == 0) { } while(C4 == 0) { } delay_ms(30); port_b = x; return; }

Descripcin del funcionamiento: Los puertos RDO, RD1, RD2 y RD3 se estlablecen como filas FA, FB, FC y FD y se definen como salidas. Las columnas C1, C2, C3 y C4 de los puertos RD4, RD5, RD6 y RD7; estn fijadas como entradas. El proceso para censar cual pulsador ha sido activado, se basa en que se todas se filas se colocan en 1 y se empieza a sacar empezando por la primera fila un 0 y se procede ha comprobar si se activado algn pulsador desde la columna 1, por ejemplo, supongamos que se accione el pulsador 1, en este caso FA=0; if (C1 == 0){ x=1; antirebote(); } ,, dado las condiciones se asigna a la variable x el valor de 1. Se llama a la funcin antirrebote y se procede a sacar el valor de x en el puerto b, para indicar el valor en el display. El proceso de barrido del teclado continua indefinidamente.

CERRADURA ELCTRICA

El siguiente proyecto trata de accionar una salida cuando se ingrese una clave previamente almacenado en el programa. En un array de 4 nmeros se guarda los nmeros de la clave (int clave [] = {1,2,3,4}). Los datos provenientes del teclado se ingresan en el array datos[], en la funcin almacenar . Una vez que se tienen los 4 nmeros, se llama a la funcin procesar para comprobar que este de acuerdo con los de la clave. Si son correctos se llama a la funcin abrir para activar el puerto b6 que acciona la puerta. En el caso contrario a la funcin sirena activa la alarma. ///CERRADURA - TECLADO PROVISTO DE ANTIRREBOTE #include <18f4550.h> //archivo de cabecera #fuses HS,WDT,NOPROTECT,NOPUT, NOPBADEN //ordenes para el programador #use delay (clock=12000000) //Fosc=12Mhz #use fast_io(b) #use fast_io(d) #BYTE port_b= 0xf81 #BYTE port_d= 0xf83 #BIT led = 0xf81.5 //LED INDICADOR #BIT puerta = 0xf81.6 //PUERTA #BIT alarma = 0xf81.7 //ALARMA #BIT FA = 0xf83.0 #BIT FB = 0xf83.1 #BIT FC = 0xf83.2 #BIT FD = 0xf83.3 #define C1 bit_test(port_d,4) #define C2 bit_test(port_d,5) #define C3 bit_test(port_d,6) #define C4 bit_test(port_d,7) #define num 4 void antirebote(); void almacenar(); void procesar(); void sirena(); void abrir(); int i; //para contar el nmero de veces que repite la alarma int x; //toma el valor de la tecla const int clave[]={1,2,3,4};//Almacena clave int n;// para contar el nmero de datos que ingresan int datos[num];//almacena el nmero de la tecla ///PROGRAMA; void main() { set_tris_b(0x00);//Puerto b como salida

set_tris_d(0xf0); //Puerto d 4 primeros bits como salida y 4 bits como entrada. disable_interrupts(GLOBAL); //todas las interrupciones desactivadas port_b = 0; FA=1; FB=1; FC=1; FD=1; n= 1; while(TRUE){ //BARRIDO FA=0; if (C1 == 0){ x=1; antirebote(); } if (C2 == 0){ x=2; antirebote(); } if (C3 == 0){ x=3; antirebote(); } if (C4 == 0){ x=10; antirebote(); } FA=1; // FB=0; if (C1 == 0){ x=4; antirebote(); } if (C2 == 0){ x=5; antirebote(); } if (C3 == 0){ x=6; antirebote(); } if (C4 == 0){ x=11; antirebote(); } FB=1;

// FC=0; if (C1 == 0){ x=7; antirebote(); } if (C2 == 0){ x=8; antirebote(); } if (C3 == 0){ x=9; antirebote(); } if (C4 == 0){ x=12; antirebote(); } FC=1; // FD=0; if (C1 == 0){ x=13; antirebote(); } if (C2 == 0){ x=0; antirebote(); } if (C3 == 0){ x=14; antirebote(); } if (C4 == 0){ x=15; antirebote(); } FD=1; }//bucle infinito } //FUNCIN ANTIRREBOTE void antirebote() { while(C1 == 0) { } // No realiza nada hasta que el pulsador est inactivo while(C2 == 0) { } while(C3 == 0) { } while(C4 == 0) { }

delay_ms(30); port_b = x; almacenar(); return; } //ALMACENAR DATOS DEL TECLADO void almacenar() { if (n <= 4){ datos[n]= x; n= n + 1; } if(n==5){ procesar(); } return; } //COMAPARA DATOS INGRESADOS CON LOS ESTABLECIDOS void procesar() { n = 1; if (datos[1]== clave[0] && datos[2]== clave[1] && datos[3]== clave[2] && datos[4]== clave[3]) abrir(); else sirena(); return; } //ABRE LA PUERTA void abrir() { puerta=1; delay_ms(5000); puerta=0; return; } //SUENA LA ALARMA void sirena() { for (i=1;i<=10; ++i) { alarma=1; delay_ms(500); alarma=0;

delay_ms(500); } return; }

También podría gustarte