Chào các bạn trong bài đăng này mình sẽ hướng dẫn các bạn về Counter của Timer 1 của Vi Điều Khiển PIC16F887 dử dụng trình biên dịch Mplab IDE và XC8 để viết và biện dịch code.
1. Hoạt động của Timer1 ở chế độ Counter đồng bộ
Khi bit TMR1CS bằng 1 thì T1 hoạt động ở chế độ Counter:
đồng bộ, nếu CPU ở chế độ ngủ thì Timer1 sẽ không đếm vì mạch đồng bộ ngừng hoạt động.
2. Hoạt động của Timer1 ở chế độ Counter bất đồng bộ
- Nếu bit T1SYNC bằng 1 thì xung ngõ vào từ bên ngoài không được đồng bộ. Bộ đếm tiếp tục
tăng bất đồng bộ với xung bên trong. Bộ đếm vẫn đếm khi CPU ở trong chế độ ngủ và khi tràn sẽ phát sinh ngắt và đánh thức CPU. Ngắt T1 có thể ngăn được.
- Để sử dụng chế độ Counter của Timer 1 chúng ta chỉ cần cấu hình các thanh ghi và bit dưới đây là chúng ta có thể sử dụng chế độ Counter của Timer 1 và các thanh ghi và bit cần cấu hình như sau :
T1CONbits.TMR1CS = 1 : Chọn nguồn xung clock từ bên ngoài ở chân RC0/T1OSO/T1CKI (cạnh lên).
T1CONbits.T1SYNC = 0 : Đồng bộ ngõ vào clock ở từ bên ngoài.
T1CONbits.TMR1ON = 1: Cho phép Timer1 đếm.
- Ngoài việc cấu hình các thanh ghi và bit ở trên chúng ta cần quan tâm tới 1 việc quan trọng nữa là cấp nguồn Clock và chân RC0 thì khi có cạnh lên hoặc cạnh xuống thì chế độ Counter sẽ bắt đầu đếm.
- Đây là ảnh mô phỏng protues.
1. Hoạt động của Timer1 ở chế độ Counter đồng bộ
Khi bit TMR1CS bằng 1 thì T1 hoạt động ở chế độ Counter:
- Nếu bit T1OSCEN bằng 1 thì đếm xung từ mạch dao động của T1.
- Nếu bit T1OSCEN bằng 1 thì đếm xung cạnh lên đưa đến ngõ vào RC0/T1OSO/T1CKI.
đồng bộ, nếu CPU ở chế độ ngủ thì Timer1 sẽ không đếm vì mạch đồng bộ ngừng hoạt động.
2. Hoạt động của Timer1 ở chế độ Counter bất đồng bộ
- Nếu bit T1SYNC bằng 1 thì xung ngõ vào từ bên ngoài không được đồng bộ. Bộ đếm tiếp tục
tăng bất đồng bộ với xung bên trong. Bộ đếm vẫn đếm khi CPU ở trong chế độ ngủ và khi tràn sẽ phát sinh ngắt và đánh thức CPU. Ngắt T1 có thể ngăn được.
- Để sử dụng chế độ Counter của Timer 1 chúng ta chỉ cần cấu hình các thanh ghi và bit dưới đây là chúng ta có thể sử dụng chế độ Counter của Timer 1 và các thanh ghi và bit cần cấu hình như sau :
T1CONbits.TMR1CS = 1;
T1CONbits.T1SYNC = 0;
T1CONbits.TMR1ON = 1;
Giải thích về 3 thanh ghi và bit ở trên như sau :T1CONbits.T1SYNC = 0;
T1CONbits.TMR1ON = 1;
T1CONbits.TMR1CS = 1 : Chọn nguồn xung clock từ bên ngoài ở chân RC0/T1OSO/T1CKI (cạnh lên).
T1CONbits.T1SYNC = 0 : Đồng bộ ngõ vào clock ở từ bên ngoài.
T1CONbits.TMR1ON = 1: Cho phép Timer1 đếm.
- Ngoài việc cấu hình các thanh ghi và bit ở trên chúng ta cần quan tâm tới 1 việc quan trọng nữa là cấp nguồn Clock và chân RC0 thì khi có cạnh lên hoặc cạnh xuống thì chế độ Counter sẽ bắt đầu đếm.
- Đây là ảnh mô phỏng protues.
- Đây là code chương trình.
#include <stdio.h>
#include <stdlib.h>
#define _XTAL_FREQ 8000000
#include <xc.h>
// CONFIG1
#pragma config FOSC = HS // Oscillator Selection bits (HS oscillator: High-speed crystal/resonator on RA6/OSC2/CLKOUT and RA7/OSC1/CLKIN)
#pragma config WDTE = OFF // Watchdog Timer Enable bit (WDT disabled and can be enabled by SWDTEN bit of the WDTCON register)
#pragma config PWRTE = OFF // Power-up Timer Enable bit (PWRT disabled)
#pragma config MCLRE = OFF // RE3/MCLR pin function select bit (RE3/MCLR pin function is digital input, MCLR internally tied to VDD)
#pragma config CP = OFF // Code Protection bit (Program memory code protection is disabled)
#pragma config CPD = OFF // Data Code Protection bit (Data memory code protection is disabled)
#pragma config BOREN = OFF // Brown Out Reset Selection bits (BOR disabled)
#pragma config IESO = OFF // Internal External Switchover bit (Internal/External Switchover mode is disabled)
#pragma config FCMEN = OFF // Fail-Safe Clock Monitor Enabled bit (Fail-Safe Clock Monitor is disabled)
#pragma config LVP = OFF // Low Voltage Programming Enable bit (RB3 pin has digital I/O, HV on MCLR must be used for programming)
// CONFIG2
#pragma config BOR4V = BOR40V // Brown-out Reset Selection bit (Brown-out Reset set to 4.0V)
#pragma config WRT = OFF // Flash Program Memory Self Write Enable bits (Write protection off)
unsigned int dem = 0,nghin,tram,chuc,donvi;
const unsigned char maled[] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F};
void display(void);
void main(void)
{
T1CONbits.TMR1CS = 1;
T1CONbits.T1SYNC = 0;
T1CONbits.TMR1ON = 1;
TRISB = 0;
PORTB = 0;
TRISDbits.TRISD0 = 0;
TRISDbits.TRISD1 = 0;
TRISDbits.TRISD2 = 0;
TRISDbits.TRISD3 = 0;
PORTD = 0;
while (1)
{
display();
dem=TMR1;
if(dem>=1500)
{
TMR1=0;
dem=0;
}
}
}
void display (void)
{
nghin = dem/1000;
tram = (dem - nghin*1000)/100;
chuc = (dem - nghin*1000 - tram*100)/10;
donvi = dem - nghin*1000 - tram*100 - chuc*10;
PORTB = maled[nghin];
RD0=0;
__delay_ms(5);
RD0=1;
PORTB = maled[tram];
RD1=0;
__delay_ms(5);
RD1=1;
PORTB = maled[chuc];
RD2=0;
__delay_ms(5);
RD2=1;
PORTB = maled[donvi];
RD3=0;
__delay_ms(5);
RD3=1;
}
EmoticonEmoticon