.................................................. ....................
Thông báo
Collapse
No announcement yet.
Công cụ học tập và phát triển cho PIC - Hoàn toàn miện phí
Collapse
This is a sticky topic.
X
X
-
Quang Nhat
---------------------------------------
Yahoo :quangnhat85ls
Mail :
Nhận thiết kế và ép nhựa cho đồ điện tử
Comment
-
Bạn nêu bài toán không rõ ràng. Người đọc khó mà hiểu được. Đưa xung vào ngắt ngoài thì bắt mức cao hay mức thấp?
Nếu bạn muốn xuất xung có mức thấp 0.4ms thì bạn có thể dùng ngắt Timer để tạo hoặc cách đơn giản là dùng hàm delay. Khi dùng timer thì tín hiệu để start cho timer lấy chính từ ngắt ngoài của PIC.Ethernet-RS232, PIC Webserver, RFID Reader
CallerID, Cảnh báo BTS, ...
0988006696
linhnc308@gmail.com
http://linhnc308.blogspot.com
Comment
-
hoi trong capture
#include <16F877A.h>
#include <def_877a.h>
#use delay(clock=20000000)
#FUSES NOWDT, HS, NOPUT, NOPROTECT, NODEBUG, NOBROWNOUT, NOLVP, NOCPD, NOWRT
#use rs232(baud=9600,parity=N,xmit=PIN_C6,rcv=PIN_C7,bi ts=9)
#bit TMR1IF = 0x0C.0em chưa hiểu lệnh này ạ?
#include <LCD_lib_4bit.c>
int16 CCP1Value; // Gia tri CCP hien tai
int16 CCP1OldValue; // Gia tri CCP truoc do
BOOLEAN CCP1Captured; --> khai báo hàm capture ạ?
float Freq;
int8 char1,char2,char3,char4,char5,char6,mode;
int8 count,count1,count2,count3;
#int_CCP1
CCP1_isr()
{
if(TMR1IF)
{
CCP1Value = CCP_1 +(65535-CCP1OldValue); sao anh lại có công thức này nhỉ?
CCP1OldValue = CCP_1; ???
TMR1IF=0; ??????????
}
else
{
CCP1Value = CCP_1 - CCP1OldValue;
CCP1OldValue = CCP_1;
}
CCP1Captured = TRUE;
}
//--------------------------------------------------------------------------
#INT_TIMER0
Timer0_isr()
{
}
//----------------------------------------
void Init_ccp(void)
{
setup_ccp1(CCP_CAPTURE_RE);
setup_timer_0(RTCC_EXT_L_TO_H|RTCC_DIV_64);
setup_timer_1(T1_INTERNAL|T1_DIV_BY_8);
set_timer0(0);
CCP1Value = 0;
CCP1OldValue = 0;
CCP1Captured = TRUE;
enable_interrupts(INT_CCP1);
//enable_interrupts(INT_TIMER1);
enable_interrupts(GLOBAL);
}
//--------------------------------------------------------------------------
void Convert_CCP1()
{
int32 temp;
//Freq = (1.0/((float)CCP1Value*2e-7)); // For Time_Div_1
Freq = (1.0/((float)CCP1Value*16e-7)); //For_Time_Div_8
if (Freq >= 1000 )
{
mode = 1;
temp = freq;
char1 = (temp / 100000) + 0x30;
temp = temp % 100000;
char2 = (temp / 10000) + 0x30;
temp = (temp % 10000);
char3 = (temp / 1000) + 0x30;
temp = (temp % 1000);
char4 = (temp / 100) + 0x30;
temp = temp % 100;
char5= (temp/10 ) + 0x30;
char6 = (temp % 10) + 0x30;
goto exit;
}
else
if (Freq >= 1000000 )
{
mode = 2;
temp = freq;
char1 = (temp / 100000) + 0x30;
temp = temp % 100000;
char2 = (temp / 10000) + 0x30;
temp = (temp % 10000);
char3 = (temp / 1000) + 0x30;
temp = (temp % 1000);
char4 = (temp / 100) + 0x30;
temp = temp % 100;
char5= (temp/10 ) + 0x30;
char6 = (temp % 10) + 0x30;
goto exit;
}
else
{
mode = 0;
temp = (int32)(freq * 1000);
char1 = (temp / 100000) + 0x30;
temp = temp % 100000;
char2 = (temp / 10000) + 0x30;
temp = (temp % 10000);
char3 = (temp / 1000) + 0x30;
temp = (temp % 1000);
char4 = (temp / 100) + 0x30;
temp = temp % 100;
char5= (temp/10 ) + 0x30;
char6 = (temp % 10) + 0x30;
goto exit;
}
exit:
temp = 0;
}
//--------------------------------------------
void convert_timer0_value()
{
count = get_timer0();
count1 = count/100 + 0x30;
count = count%100;
count2 = count/10 + 0x30;
count3 = count%10 + 0x30;
}
//---------------------------------------------------------------
void main()
{
TRISB = 0; // F = 1/T
Init_ccp(); // Timer1 prescaler DIV_BY_1
LCD_init(); // PIC16f877a 20MHz -> 0.0000002 = 200nS
//printf("Frequence test:\r\n"); // Pic16F877A 4MHz -> 0.000001 = 1uS
LCD_putcmd(0xC0);
Printf(LCD_putchar,"Counter = ");
LCD_putcmd(0x80);
Printf(LCD_putchar,"Freq = ");
while (TRUE)
{
if (CCP1Captured)
{
Convert_CCP1();
LCD_putcmd(0x87);
if (char1 != 0x30) LCD_putchar(char1);
if (((char1 != 0x30) && (char2 == 0x30)) || (char2 != 0))
LCD_putchar(char2);
LCD_putchar(char3);
LCD_putchar(".");
LCD_putchar(char4);
LCD_putchar(char5);
switch (mode)
{
case 0: {LCD_putchar(char6);lcd_putcmd(0x8F);Printf(LCD_pu tchar," Hz");} break;
case 1: {LCD_putchar(char6);lcd_putcmd(0x8F);Printf(LCD_pu tchar,"KHz");} break;
case 2: {LCD_putchar(char6);lcd_putcmd(0x8F);Printf(LCD_pu tchar,"MHz");} break;
}
printf("Freq:%f\r\n",Freq);
CCP1Captured = FALSE;
}
convert_timer0_value();
LCD_putcmd(0xCa);
LCD_putchar(count1);
LCD_putchar(count2);
LCD_putchar(count3);
}
}
và tiện đây cho em hỏi luôn ạ. em đã nạp chương trình của anh chạy vào PÍC6F, dầu vào là RC2(ccp1) là SRF05 +lcd1602 nhưng gia trị hiển thị lên LCD toàn số 0, ko bít sao nhỉ?
Comment
-
Comment
-
#bit TMR1IF = 0x0C.0 -> Đây không phải là lệnh, đây là khai báo địa chỉ bit cờ ngắt Timer 1, xem datasheet sẽ thấy có bít này.
Code:if(TMR1IF) { CCP1Value = CCP_1 +(65535-CCP1OldValue); sao anh lại có công thức này nhỉ? CCP1OldValue = CCP_1; ??? TMR1IF=0; ?????????? } else { CCP1Value = CCP_1 - CCP1OldValue; CCP1OldValue = CCP_1; }
Như vậy nếu tại sườn 1, ta copy giá trị này vào CCP1OldValue, đến sườn tiếp theo ta lấy hiệu CCP_1 - CCP1OldValue thì sẽ có được giá trị độ rộng xung tính theo số xung clock hệ thống mà Timer 1 đếm được, từ đó tính ra được tần số. Lưu ý là Timer 1 lúc này chạy liên tục ở chế độ 16bits, nhận clock từ hệ thống, theo hệ số chia do user đặt.
@Anh_Gioi: Em vẽ ra giản đồ thời gian xung vuông và so sánh với chu kỳ ngắt Timer 1 cho 2 trường hợp:
1. Hai sườn xung nằm trong khoảng thời gian ngắt Timer 1 -> tương ứng với lệnh trong else
2. có sườn xung thứ nhất, tiếp theo có ngắt Timer1, rồi sau mới có sườn thứ 2 -> Tương ứng với điều kiện If bằng true.
-----------______------ ______
Xung : ___|----- |_____|-------|______
Timer1: -|-----------------------|------
TMR1IF = 1
-----------______------ ______
Xung : ___|------|_____|------|______
Timer1: -|------------|-----------------Last edited by linhnc308; 13-10-2008, 10:59.Ethernet-RS232, PIC Webserver, RFID Reader
CallerID, Cảnh báo BTS, ...
0988006696
linhnc308@gmail.com
http://linhnc308.blogspot.com
Comment
-
Nguyên văn bởi linhnc308 Xem bài viết#bit TMR1IF = 0x0C.0 -> Đây không phải là lệnh, đây là khai báo địa chỉ bit cờ ngắt Timer 1, xem datasheet sẽ thấy có bít này.
Code:if(TMR1IF) { CCP1Value = CCP_1 +(65535-CCP1OldValue); sao anh lại có công thức này nhỉ? CCP1OldValue = CCP_1; ??? TMR1IF=0; ?????????? } else { CCP1Value = CCP_1 - CCP1OldValue; CCP1OldValue = CCP_1; }
Như vậy nếu tại sườn 1, ta copy giá trị này vào CCP1OldValue, đến sườn tiếp theo ta lấy hiệu CCP_1 - CCP1OldValue thì sẽ có được giá trị độ rộng xung tính theo số xung clock hệ thống mà Timer 1 đếm được, từ đó tính ra được tần số. Lưu ý là Timer 1 lúc này chạy liên tục ở chế độ 16bits, nhận clock từ hệ thống, theo hệ số chia do user đặt.
@Anh_Gioi: Em vẽ ra giản đồ thời gian xung vuông và so sánh với chu kỳ ngắt Timer 1 cho 2 trường hợp:
1. Hai sườn xung nằm trong khoảng thời gian ngắt Timer 1 -> tương ứng với lệnh trong else
2. có sườn xung thứ nhất, tiếp theo có ngắt Timer1, rồi sau mới có sườn thứ 2 -> Tương ứng với điều kiện If bằng true.
-----------______------ ______
Xung : ___|----- |_____|-------|______
Timer1: -|-----------------------|------
TMR1IF = 1
-----------______------ ______
Xung : ___|------|_____|------|______
Timer1: -|------------|-----------------
RC2--input ECHO
D1 - output trigger
#include <16F877A.h>
#include <F:\NCKH 2009\PIC 16\count_timer2\def_877a.h>
#use delay(clock=2000000)
#fuses HS,NOWDT, NOPROTECT
#include <F:\NCKH 2009\PIC 16\count_timer2\LCD_lib_4bit.c>
int16 CCP1Value;
int16 CCP1OldValue;
BOOLEAN CCP1Captured;
float S,count;
int8 char1,char2,char3;
#int_CCP1 // Ngat do CCCP1 xay ra, thuc hien lenh...
CCP1_isr()
{
CCP1Value = CCP_1 - CCP1OldValue;
CCP1OldValue = CCP_1;
CCP1Captured = TRUE;
}
//--------------------------------------------------------------------------
void Init_ccp(void) //Khoi tao chuc nang CCCP
{
setup_ccp1(CCP_CAPTURE_RE);
setup_timer_1(T1_INTERNAL|T1_DIV_BY_8);
CCP1Value = 0;
CCP1OldValue = 0;
CCP1Captured = TRUE;
enable_interrupts(INT_CCP1);
enable_interrupts(GLOBAL);
}
//--------------------------------------------------------------------------
void Convert_CCP1()
{
long temp;
int8 temp2;
//----- thoi gian SRF doc duoc
count = CCP1Value*1.6e-6;
// Timer1 prescaler DIV_BY_8
//Freq = 1.0/((float)CCP1Value*8e-6);
//-------- khoang cach can do--
S = count/58;
//--------chuyen sang BCD----
temp = (long)S;
char1 = ((temp / 100) + 0x30);
temp2 = (temp % 100);
char2 = ((temp2 / 10) + 0x30);
char3 = ((temp2 % 10) + 0x30);
//-----hien thi LCD----
LCD_putcmd(0x80);
LCD_putchar("S = ");
LCD_putcmd(0x8B);
LCD_putchar(" Cm");
LCD_putcmd(0x84);
LCD_putchar(char1);
LCD_putcmd(0x85);
LCD_putchar(char2);
LCD_putcmd(0x86);
LCD_putchar(char3);
LCD_putcmd(0xC5);
LCD_putchar("Anh_Gioi");
}
void main()
{
Init_ccp();
LCD_init();
while (TRUE) {
//---- 10uS cho trgger ----
output_high(pin_d1);
delay_us(10);
output_low(pin_d1);
if (CCP1Captured)
Convert_CCP1();
}
}
cuối tuần này em phải nộp cho thầy rùi mà vẫn chưa xong...Hi vọng có sự hỗ trợ tiếp từ bên các anh !
Comment
-
thay thế count = CCP1Value*1.6e-6; bởi count = CCP1Value; thôi để check xem giá trị của CCP1 là bao nhiêu, vì khi em thưc hiện phép nhân với 1.6e-6 sẽ có giá trị rất nhỏ. Nên tính toàn lại cho hàm convert để có giá trị hiển thị đúng. Chú ý giá trị timer của em là DIV_8. Do đó khi tính lại giá trị cho Count, em phải tính lại và phải dùng đến giá trị thạch anh 20MHz.
1 gia tri dem cua timer bang tan so he thong chia cho 8, nhu vay tan so cua timer la 20M/8 -> tu do, ung voi gia tri cua Timer thi se tinh ra duoc tan so dau vao hay thoi gian giua 2 xung.Ethernet-RS232, PIC Webserver, RFID Reader
CallerID, Cảnh báo BTS, ...
0988006696
linhnc308@gmail.com
http://linhnc308.blogspot.com
Comment
-
Nguyên văn bởi linhnc308 Xem bài viếtthay thế count = CCP1Value*1.6e-6; bởi count = CCP1Value; thôi để check xem giá trị của CCP1 là bao nhiêu, vì khi em thưc hiện phép nhân với 1.6e-6 sẽ có giá trị rất nhỏ. Nên tính toàn lại cho hàm convert để có giá trị hiển thị đúng. Chú ý giá trị timer của em là DIV_8. Do đó khi tính lại giá trị cho Count, em phải tính lại và phải dùng đến giá trị thạch anh 20MHz.
1 gia tri dem cua timer bang tan so he thong chia cho 8, nhu vay tan so cua timer la 20M/8 -> tu do, ung voi gia tri cua Timer thi se tinh ra duoc tan so dau vao hay thoi gian giua 2 xung.
bây giờ em phải tính lại thời gian,em dùng thạch anh 20M thì công thức thời gian đọc được
Freq = 1.0/((float)CCP1Value*8e-6);
là đúng mà anh????
như anh nói thì như vậy thời gian =20M/8*giá trị timer;
quãng đường em do được S = thòi gian/58(cm) cóphải không ạ?Last edited by anh_gioi; 15-10-2008, 22:07.
Comment
-
Em đang làm đề tài tốt nghiệp về PIC_Internet anh có thể cho em xin cái code và mạch Layout được ko?Tối qua vô trang web kia www.Olimex.com mà ko tìm thấy code.Giúp em cái.Em cảm ơn!
Comment
Bài viết mới nhất
Collapse
-
Trả lời cho tìm mua đồng hồ vạn năngbởi mèomướpDạ chú mua cái kẹp dòng ấy ạ. Chị hàng xóm nhà cháu có 1 cái thấy lâu lâu rùi chưa hỏng ạ. Ví dụ như mẫu này trên shoppee đầy ạ...
https://vn.shp.ee/dWYVgq7-
Channel: Điện tử dành cho người mới bắt đầu
Hôm qua, 12:48 -
-
Trả lời cho Sửa bộ nguồn DC 60V 45Abởi vi van phamBác Đinh Vặn sai rồi,bây giờ con nít mẫu giáo đã giải phương trình 2 ẩn số rồi.
Tôi chứng minh bác lên youtube đầy video đơn giản tựa rất hot, chỉ 1 transistor hay 1 con diode và hướng dẩn cách làm, tác giả không vẽ sơ đồ mạch điện...-
Channel: Điện tử công suất
Hôm qua, 11:55 -
-
Trả lời cho Sửa bộ nguồn DC 60V 45Abởi nguyendinhvanBây giờ mới có tháng giêng, bao giờ mới đến tháng mười ?
Các cháu mẫu giáo mới lên lớp 1 được có 4 tháng. Nên đừng lấy lý do chúng nó đã đi học rồi để bắt nó viết một bài luận văn, hay là giải bài toán hàm. Phải kiên nhẫn...-
Channel: Điện tử công suất
Hôm qua, 00:47 -
-
Trả lời cho tìm mua đồng hồ vạn năngbởi nguyendinhvanKhông có loại nào đủ một vạn chức năng đâu. Nó chỉ được 2345678 , hoặc khủng lắm thì được 10 chức năng.
-
Channel: Điện tử dành cho người mới bắt đầu
Hôm qua, 00:16 -
-
bởi since0501Cảm ơn bác đã chia sẻ!...
-
Channel: Điện tử dành cho người mới bắt đầu
19-01-2025, 19:00 -
-
bởi Lê Gia TứMình muốn tìm mua đồng hồ vạn năng giá khoảng 200k có đo tần số cao khoảng 0~1mhz mọi người tư vấn giúp mình với
-
Channel: Điện tử dành cho người mới bắt đầu
19-01-2025, 15:47 -
-
Trả lời cho Mạch tự động bật nguồnbởi davidcopyChỉ cần dùng R C mắc vô phím power là ok....
-
Channel: Điện tử gia dụng
18-01-2025, 20:47 -
-
Trả lời cho Hỏi cách điều chế xungbởi davidcopydùng mach khuếch opamp...
-
Channel: Kỹ thuật điện tử tương tự
18-01-2025, 20:42 -
-
Trả lời cho Thắc mắc về hạ áp cho adapter laptopbởi davidcopy
-
Channel: Điện tử dành cho người mới bắt đầu
18-01-2025, 18:56 -
-
bởi bqvietĐấy gọi là cái "điểm gãy" trong đồ thị điện áp - dung lượng còn lại. Dùng điện áp hở cực để xác định sắp hết hoặc gần đầy thì vẫn tạm ổn. Dùng để đo SOC/DOD thì đừng. Ngay cả số km đã đi cũng chả xác định được...
-
Channel: Điện tử dành cho người mới bắt đầu
17-01-2025, 21:36 -
Comment