![]() |
|
Tài trợ cho PIC Vietnam |
Cơ bản về vi điều khiển và PIC Những bài hướng dẫn cơ bản nhất để làm quen với vi điều khiển PIC |
![]() |
|
Ðiều Chỉnh | Xếp Bài |
![]() |
#1 |
Đệ tử 6 túi
Tham gia ngày: Oct 2008
Nơi Cư Ngụ: HCM
Bài gửi: 137
: |
![]() setup_timer_2 (mode, period, postscale)
mode :T2_DISABLED, T2_DIV_BY_1, T2_DIV_BY_4, T2_DIV_BY_16 period : int 0-255 postscale :1-16 bác nào bít cho mình hỏi 3 cái option trên có ý nghĩa như thế nào vậy? nếu mình muốn ngắt timer2 xảy ra sau 1s thì mình phải set như thế này phải không(thạch anh 20M)? setup_timer_2(t2_div_by_1,56,16) #int_timer2 void int_timer2_isr() { timer2++; if(timer2==25000) { count++; timer2=0; } } biến count tăng 1 đơn vị sau 1s. |
![]() |
![]() |
![]() |
#2 |
Đệ tử 6 túi
Tham gia ngày: Nov 2008
Nơi Cư Ngụ: Di Linh
Bài gửi: 131
: |
Timer 2 có 8 bit, bộ chia trước , bộ chia sau.
setup_timer_2(t2_div_by_1,56,16) --> chia trước 1, chia sau 16 ( timer tràn 16 lần thì xảy ra ngắt ). prescale = 1 : tức timer tăng sau 1 ckm = 0.2 us ( vì Xtal = 20 Mhz). Giá trị ban đầu bạn đặt là 56 --> thời gian 1 lần tràn là 200 x 0.2 = 40 us . postscale = 16 : timer tràn 16 lần thì xảy ra ngắt. Vậy tổng thời gian của 1 lần ngắt là : 40 x 16 = 640 us , quá nhỏ so với ý định 1s của bạn |
![]() |
![]() |
![]() |
#3 | |
Đệ tử 6 túi
Tham gia ngày: Oct 2008
Nơi Cư Ngụ: HCM
Bài gửi: 137
: |
Trích:
thực ra là mình muốn sau 40us thì xảy ra ngắt và tăng biến timer2 lên 1 đơn vị và sau 25000 lần ngắt (40us*25000=1000000us=1s)thì biến count mới tăng lên 1 đơn vị. Như vậy để được như mong muốn thì mình phải làm như thế này phải không bạn: setup_timer_2(t2_div_by_1,56,1); Nhưng mình không hiểu sao là khi mình làm như vậy thì mình có cảm giác pic bị reset liên tục, vì các chương trình quét led bị nhấp nháy liên tục. |
|
![]() |
![]() |
![]() |
#4 |
Đệ tử 6 túi
Tham gia ngày: Nov 2008
Nơi Cư Ngụ: Di Linh
Bài gửi: 131
: |
Nếu bạn muốn timer2 ngắt sau 1 s thì bạn có thể làm thế này:
XTAL = 4 Mhz, 1 ckm = 1 us setup_timer_2(t2_div_by_16,6,10); // 250 x 16 x 10 = 40.000 us ....... #int_timer2 // ngắt timer 2 xảy ra void int_timer2_isr() { set_timer2(56) ; // nạp lại cho timer 2 i++ ; // tăng biến đếm if ( i == 25 ) /// 25 x 4.000 = 1.000.000 us { thực thi điều gì khi đủ 1 s |
![]() |
![]() |
![]() |
#5 |
Đệ tử 6 túi
Tham gia ngày: Nov 2008
Nơi Cư Ngụ: Di Linh
Bài gửi: 131
: |
xí tì 1 chút. Nhắc lại timer2 tự nhiên ngồi nhớ lại. Mục 7.1 trang 56 của data sheet PIC 16F87X có nói :
7.1 Timer2 Prescaler and Postscaler The prescaler and postscaler counters are cleared when any of the following occurs: • a write to the TMR2 register • a write to the T2CON register • any device Reset (POR, MCLR Reset, WDT Reset or BOR) TMR2 is not cleared when T2CON is written. ///////////////////////////////////////////////////////// Tức là các bộ chia trước, chia sau sẽ bị xóa khi sau thực thi: ghi vào t.ghi TMR2, T2CON , Reset. Vì vậy bạn phải cẩn thận khi nạp lai giá tri cho timer2, không khéo các bộ chia sẽ bị xóa. Đấy là nhận định cá nhân của mình. Các bác thấy thế nào xin cho em ý kiến. thanks |
![]() |
![]() |
![]() |
#6 | |
Đệ tử 6 túi
Tham gia ngày: Oct 2008
Nơi Cư Ngụ: HCM
Bài gửi: 137
: |
Trích:
còn nữa nếu nạp lại cho timer2 thì mình làm như bác nói: set_timer2(56) ; hay là : set_timer2(t2_div_by_16,6,10); |
|
![]() |
![]() |
![]() |
#7 |
Đệ tử 6 túi
Tham gia ngày: Nov 2008
Nơi Cư Ngụ: Di Linh
Bài gửi: 131
: |
uh sorry nạp lại cho timer2 là 6 chứ ko phải 56 nha bạn.
Dù trong CCS không có nói về việc bộ chia có bị ảnh hưởng ntn khi thực thi lệnh set_timer2(6), nhưng tốt nhất bạn nên dùng lệnh set_timer2(t2_div_by_16,6,10). |
![]() |
![]() |
![]() |
#8 |
Đệ tử 6 túi
Tham gia ngày: Oct 2008
Nơi Cư Ngụ: HCM
Bài gửi: 137
: |
Đã kiểm soát được timer2 rùi!Cảm ơn bác phandinh nhiều nhiều lắm!
|
![]() |
![]() |
![]() |
#9 |
Nhập môn đệ tử
Tham gia ngày: Apr 2009
Bài gửi: 1
: |
Theo mình hiều thì cách tính thời gian tràn và ngắt của timer2 ko như timer0 và timer1.
TD trên của phandinh: XTAL = 4 Mhz, 1 ckm = 1 us setup_timer_2(t2_div_by_16,6,10); // 250 x 16 x 10 = 40.000 us Theo mình thì, timer2 sẽ tăng mỗi 4/(4*10^6)*16=16us, thời gian tràn sẽ là (6+1)*16us=112us, thời gian ngắt 112us*10=1120us!!! |
![]() |
![]() |
![]() |
#10 |
Nhập môn đệ tử
Tham gia ngày: Oct 2009
Bài gửi: 1
: |
chính xác là như thế.timer2 ngắt là sau 1,12ms.
|
![]() |
![]() |
![]() |
#11 | |
Đệ tử 1 túi
Tham gia ngày: Apr 2010
Bài gửi: 16
: |
Trích:
setup_timer_2(t2_div_by_16,6,10); // 250 x 16 x 10 = 40.000 us Em thấy cái này đúng rồi mà. Còn cái bên dưới này: Theo mình thì, timer2 sẽ tăng mỗi 4/(4*10^6)*16=16us, thời gian tràn sẽ là (6+1)*16us=112us, thời gian ngắt 112us*10=1120us!!! thì sao e hok hiểu, ai giải thích giúp e với. |
|
![]() |
![]() |
![]() |
#12 | |
Đệ tử 1 túi
Tham gia ngày: Apr 2010
Bài gửi: 16
: |
Trích:
|
|
![]() |
![]() |
![]() |
#13 |
Nhập môn đệ tử
Tham gia ngày: Aug 2011
Bài gửi: 1
: |
Đến giờ chắc Alycuong đã hiểu vấn đề timer2 rồi nhưng mình cũng xin viết vài dòng để kết luôn.
Giải thích thứ 2 theo mình là đúng. Cách tính như sau: - T2_div_by_16 : 1us x 16 = 16us - timer tăng lên mỗi 16us - 6: 16us x(6+1) số lần = 112 - cờ tràn sau mỗi 112us - 10: 10 x 112 = 1120us - ngắt sau mỗi 1120us Trong tài tiệu CCS giải thích rõ lắm |
![]() |
![]() |
![]() |
#14 |
Đệ tử 1 túi
Tham gia ngày: Mar 2012
Bài gửi: 16
: |
giải thích thứ 2 mới là chính xác
|
![]() |
![]() |
![]() |
#15 |
Đệ tử 1 túi
Tham gia ngày: Apr 2010
Bài gửi: 12
: |
setup_timer_2 (mode, period, postscale)
// set cho bộ chia trước mode :T2_DISABLED, T2_DIV_BY_1, T2_DIV_BY_4, T2_DIV_BY_16 // set khoảng tràn cho timer2, khác các timer khác nghe. // vd bạn set là period=5 thì sau 5+1(set cờ báo tràn) xung nhịp nó tràn period : int 0-255 // set cho bộ chia sau postscale :1-16 để tạo khoảng định thời 1S các bạn tham khảo code sau: #include <16F877A.h> #FUSES NOWDT //No Watch Dog Timer #FUSES hs //High speed Osc (> 4mhz for PCM/PCH) (>10mhz for PCD) #FUSES NOBROWNOUT //No brownout reset #FUSES NOLVP //No low voltage prgming, B3(PIC16) or B5(PIC18) used for I/O #use delay(clock=20000000) #define output PIN_A1 static int count=0; void main() { setup_timer_2(T2_DIV_BY_16, 0x06, 10); enable_interrupts(GLOBAL); enable_interrupts(INT_TIMER2); while(true); } #INT_TIMER2 void fulse_1s(){ setup_timer_2(T2_DIV_BY_16, 249, 16); // timer2 tang len 1 sau 0.2*16=3.2us // tran sau 3.2*(249+1)= 800us // xay ra ngat sau 800*16= 12800us count++; if(count==78){ // 1000000/12800=78.125 output_toggle(output); count=0; } } |
![]() |
![]() |
![]() |
|
|