PIC Vietnam

Go Back   PIC Vietnam > Microchip PIC > Cơ bản về vi điều khiển và PIC

Tài trợ cho PIC Vietnam
Trang chủ Đăng Kí Hỏi/Ðáp Thành Viên Lịch Tìm Kiếm Bài Trong Ngày Ðánh Dấu Ðã Ðọc Vi điều khiển

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

Trả lời
 
Ðiều Chỉnh Xếp Bài
Old 19-07-2008, 08:48 PM   #1
namqn
Trưởng lão PIC bang
 
Tham gia ngày: Feb 2006
Nơi Cư Ngụ: Tp. HCM, Việt Nam
Bài gửi: 3,025
:
Send a message via Yahoo to namqn
Trích:
Nguyên văn bởi bachelor View Post
Thanhks, như vậy là đã rõ về hàm set_timerX();
Vấn đề tiếp theo
Trở lại bài toán mà tôi đã đề cập ở trên, tuy nhiên khi này khoảng cách (thời gian) giữa 2 xung thay đổi trong một dải lớn, giả thiết là từ 100us đến 10ms.
Vẫn sử dụng timer0 làm bộ định thời với clock nội = 1MHz, prescale = 1.
Khi này không thể sử dụng cách mà tôi đã nêu ở bài trước (vì với những khoảng thời gian > 256us thì rõ ràng hàm get_timer0() sẽ trả về giá trị sai)
Trong trường hợp này phương án giải quyết của tôi như sau:
- ở phần khởi tạo, cho phép ngắt bởi bộ định thời 0.
- phần chương trình thực hiện bởi ngắt định thời 0: tăng biến đếm i lên 1 (i là biến toàn cục)
- ngắt ngoài khi có xung thứ nhất đến: gán i = 0; gọi hàm set_timer0(0); gọi hàm enable_interrupts(INT_TIMER0);
- ngắt ngoài khi có xung thứ hai đến: diasble_interrupts(INT_TIMER0); Thời gian cần đo = i*256 + get_timer0(); [đơn vị us]
Khi test chương trình trên proteus với khoảng cách xung lớn thì khá chính xác, nhưng khi khoảng cách nhỏ hơn 256 us thì kết quả ra không chính xác và thay đổi loạn cả lên, không hiểu vì sao
Vậy mong mọi người xem qua phương án giải quyết bài toán của tôi và chỉ giùm tôi chỗ sai sót.
To Namqn: Trên thực tế bài toán tôi cần giải quyết cho phép sai số 1 vài us khi đo. Cũng lâu rồi không vào diễn đàn, quả thật đã từng hỏi vấn đề này nhưng trước đó tôi hỏi theo khía cạnh khác có thể nói là không liên quan . Thanks !
Về mặt nguyên tắc, phương pháp thực hiện của bạn là ổn. Trong thực tế, có những yếu tố dẫn đến sai lệch kết quả đo mà bạn không đánh giá trong phương pháp trên.

Như tôi đã nói ở post trên, có thời gian trễ giữa thời điểm xuất hiện xung và việc diễn ra các thao tác. Cụ thể: khi xuất hiện xung thứ nhất, vài chu kỳ máy sau trình phục vụ ngắt ngoài mới được thực hiện, và mất thêm vài chu kỳ máy nữa cho việc lưu các thanh ghi cần được lưu (việc này trình biên dịch tự làm) rồi mới đến các lệnh thực sự. Khi xuất hiện xung thứ hai, tương tự như vậy, cũng mất một khoảng thời gian nữa mới đến việc thực thi các lệnh mà bạn muốn thực hiện cho trường hợp này, bên trong chương trình phục vụ ngắt. Thời gian trễ ở trường hợp thứ nhất và trường hợp thứ hai có thể khác nhau. Điều quan trọng hơn nữa, thứ tự thực hiện lệnh cho các trường hợp sẽ ảnh hưởng rất lớn đến sai lệch của kết quả.

Bạn nên chỉnh lại thứ tự thực hiện như sau (cho trình phục vụ ngắt ngoài):
- Ngắt ngoài khi có xung thứ nhất đến: gọi hàm set_timer0(0); gọi hàm enable_interrupts(INT_TIMER0); gán i = 0;
- Ngắt ngoài khi có xung thứ hai đến: biến_tạm = get_timer0(), diasble_interrupts(INT_TIMER0); thời gian cần đo = i*256 + biến_tạm; [đơn vị us]

Tuy nhiên, tôi cho rằng phương pháp sau tốt hơn:
1. Khởi tạo timer0 và cho phép ngắt timer0, nhưng timer0 chưa chạy; xóa thanh ghi timer0

2. Giả thiết bạn dùng PIC16/PIC18, chương trình phục vụ ngắt sẽ kiểm tra ngắt ngoài trước rồi mới kiểm tra ngắt timer0

3. Ngắt ngoài khi có xung thứ nhất đến: bật timer0 bằng hợp ngữ; gán i = 0;

4. Ngắt ngoài khi có xung thứ hai đến: đọc thanh ghi timer0 vào biến_tạm bằng hợp ngữ; tắt timer0; xóa thanh ghi timer0; thời gian cần đo = i*256 + biến_tạm; [đơn vị us]

5. Phần code phục vụ ngắt timer0 chỉ tăng i

6. Nếu thời gian đo được luôn lớn hơn hay nhỏ hơn thời gian thực tế, bạn có thể bù sai lệch này trong công thức tính thời gian cần do.

Cần nhớ: Thứ tự viết code là quan trọng, vì bạn đo thời gian, mà các lệnh cũng cần thời gian để thực thi.

Thân,
__________________
Biển học mênh mông, sức người có hạn.

Đang gặp vấn đề cần được giúp đỡ? Hãy dành ra vài phút đọc luồng sau:
http://www.picvietnam.com/forum/showthread.php?t=1263
namqn vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Old 20-07-2008, 01:28 PM   #2
bachelor
Đệ tử 2 túi
 
Tham gia ngày: Jun 2006
Bài gửi: 27
:
Trích:
Nguyên văn bởi namqn View Post
Về mặt nguyên tắc, phương pháp thực hiện của bạn là ổn. Trong thực tế, có những yếu tố dẫn đến sai lệch kết quả đo mà bạn không đánh giá trong phương pháp trên.

Như tôi đã nói ở post trên, có thời gian trễ giữa thời điểm xuất hiện xung và việc diễn ra các thao tác. Cụ thể: khi xuất hiện xung thứ nhất, vài chu kỳ máy sau trình phục vụ ngắt ngoài mới được thực hiện, và mất thêm vài chu kỳ máy nữa cho việc lưu các thanh ghi cần được lưu (việc này trình biên dịch tự làm) rồi mới đến các lệnh thực sự. Khi xuất hiện xung thứ hai, tương tự như vậy, cũng mất một khoảng thời gian nữa mới đến việc thực thi các lệnh mà bạn muốn thực hiện cho trường hợp này, bên trong chương trình phục vụ ngắt. Thời gian trễ ở trường hợp thứ nhất và trường hợp thứ hai có thể khác nhau. Điều quan trọng hơn nữa, thứ tự thực hiện lệnh cho các trường hợp sẽ ảnh hưởng rất lớn đến sai lệch của kết quả.

Bạn nên chỉnh lại thứ tự thực hiện như sau (cho trình phục vụ ngắt ngoài):
- Ngắt ngoài khi có xung thứ nhất đến: gọi hàm set_timer0(0); gọi hàm enable_interrupts(INT_TIMER0); gán i = 0;
- Ngắt ngoài khi có xung thứ hai đến: biến_tạm = get_timer0(), diasble_interrupts(INT_TIMER0); thời gian cần đo = i*256 + biến_tạm; [đơn vị us]

Tuy nhiên, tôi cho rằng phương pháp sau tốt hơn:
1. Khởi tạo timer0 và cho phép ngắt timer0, nhưng timer0 chưa chạy; xóa thanh ghi timer0

2. Giả thiết bạn dùng PIC16/PIC18, chương trình phục vụ ngắt sẽ kiểm tra ngắt ngoài trước rồi mới kiểm tra ngắt timer0

3. Ngắt ngoài khi có xung thứ nhất đến: bật timer0 bằng hợp ngữ; gán i = 0;

4. Ngắt ngoài khi có xung thứ hai đến: đọc thanh ghi timer0 vào biến_tạm bằng hợp ngữ; tắt timer0; xóa thanh ghi timer0; thời gian cần đo = i*256 + biến_tạm; [đơn vị us]

5. Phần code phục vụ ngắt timer0 chỉ tăng i

6. Nếu thời gian đo được luôn lớn hơn hay nhỏ hơn thời gian thực tế, bạn có thể bù sai lệch này trong công thức tính thời gian cần do.

Cần nhớ: Thứ tự viết code là quan trọng, vì bạn đo thời gian, mà các lệnh cũng cần thời gian để thực thi.

Thân,
Cám ơn gợi ý của namqn, sẽ nghiên cứu chèn các mã asm thích hợp để tối ưu hóa thời gian xử lý của chương trình.
Thêm một câu hỏi nữa :
1. Các xung làm nguồn ngắt ngoài có độ rộng (rỗng) 0,5 us đến 1 us. Không biết các ngắt ngoài có đáp ứng được không nhỉ (tôi chọn kích hoạt ngắt theo sườn lên). Mô phỏng trên proteus thì vẫn xử lý được, chưa thử thực tế
2. Hai xung (mà chương trình cần đo khoảng cách thời gian giữa chúng) không nằm cùng một kênh (một đường tín hiệu). Đây là lý do tôi không thể dùng timer như một counter mà phải dùng đến ngắt ngoài và sẽ phải dùng đến 2 ngắt ngoài cho 2 xung. 16f877a lại chỉ có một ngắt ngoài. Liệu có thể dùng một chân ngắt của PortB làm ngắt ngoài được không ? Lưu ý là độ rộng xung chỉ có từ 0,5 - 1 us , thằng ngắt ngoài thì còn khả dĩ nhưng liệu thằng ngắt cổng B có đáp ứng được không nhỉ ? Hay phải dùng đến PIC18 -> đắt
Hai vấn đề này có vẻ hơi lạc với chủ đề đề cập ban đầu nhưng lại là những câu hỏi mang tính thực tế Những ai đã từng có kinh nghiệm xử lý các chuỗi xung với các tham số tương tự bằng PIC16 or PIC18 hãy giúp tôi với nhé
Thanhks !
bachelor vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Old 20-07-2008, 08:26 PM   #3
namqn
Trưởng lão PIC bang
 
Tham gia ngày: Feb 2006
Nơi Cư Ngụ: Tp. HCM, Việt Nam
Bài gửi: 3,025
:
Send a message via Yahoo to namqn
Trích:
Nguyên văn bởi bachelor View Post
Cám ơn gợi ý của namqn, sẽ nghiên cứu chèn các mã asm thích hợp để tối ưu hóa thời gian xử lý của chương trình.
Thêm một câu hỏi nữa :
1. Các xung làm nguồn ngắt ngoài có độ rộng (rỗng) 0,5 us đến 1 us. Không biết các ngắt ngoài có đáp ứng được không nhỉ (tôi chọn kích hoạt ngắt theo sườn lên). Mô phỏng trên proteus thì vẫn xử lý được, chưa thử thực tế
2. Hai xung (mà chương trình cần đo khoảng cách thời gian giữa chúng) không nằm cùng một kênh (một đường tín hiệu). Đây là lý do tôi không thể dùng timer như một counter mà phải dùng đến ngắt ngoài và sẽ phải dùng đến 2 ngắt ngoài cho 2 xung. 16f877a lại chỉ có một ngắt ngoài. Liệu có thể dùng một chân ngắt của PortB làm ngắt ngoài được không ? Lưu ý là độ rộng xung chỉ có từ 0,5 - 1 us , thằng ngắt ngoài thì còn khả dĩ nhưng liệu thằng ngắt cổng B có đáp ứng được không nhỉ ? Hay phải dùng đến PIC18 -> đắt
Hai vấn đề này có vẻ hơi lạc với chủ đề đề cập ban đầu nhưng lại là những câu hỏi mang tính thực tế Những ai đã từng có kinh nghiệm xử lý các chuỗi xung với các tham số tương tự bằng PIC16 or PIC18 hãy giúp tôi với nhé
Thanhks !
1. Tôi không rõ các ngắt ngoài có thể đáp ứng với các xung đó hay không. Nhưng nếu các xung không đủ rộng thì tôi chỉ cần kéo dài các xung đó ra. Tôi ít khi mô phỏng mà chỉ dùng mạch thật để thử nghiệm cho những bài toán loại này.

2. Một cổng OR 2 ngõ vào đơn giản sẽ giúp tôi dồn 2 tín hiệu đó vào một chân ngắt ngoài. Không nhất thiết phải dùng vi mạch logic, mà vài linh kiện rời vẫn có thể thực hiện chức năng này.

Thân,
__________________
Biển học mênh mông, sức người có hạn.

Đang gặp vấn đề cần được giúp đỡ? Hãy dành ra vài phút đọc luồng sau:
http://www.picvietnam.com/forum/showthread.php?t=1263
namqn vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Old 20-07-2008, 10:26 PM   #4
bachelor
Đệ tử 2 túi
 
Tham gia ngày: Jun 2006
Bài gửi: 27
:
Trích:
Nguyên văn bởi namqn View Post
1. Tôi không rõ các ngắt ngoài có thể đáp ứng với các xung đó hay không. Nhưng nếu các xung không đủ rộng thì tôi chỉ cần kéo dài các xung đó ra. Tôi ít khi mô phỏng mà chỉ dùng mạch thật để thử nghiệm cho những bài toán loại này.

2. Một cổng OR 2 ngõ vào đơn giản sẽ giúp tôi dồn 2 tín hiệu đó vào một chân ngắt ngoài. Không nhất thiết phải dùng vi mạch logic, mà vài linh kiện rời vẫn có thể thực hiện chức năng này.

Thân,
Càng ngày càng đi chệch ra khỏi các chủ đề của diễn đàn nhưng vẫn theo luồng nên tiện hỏi luôn
1. Namqn có thể chỉ giúp một vài sơ đồ mạch kéo dãn độ rộng xung sử dụng đc trong trường hợp này (với yêu cầu sườn trước của xung sau mạch hiệu chỉnh không được giữ chậm quá nhiều so với xung đầu vào)
2. Cái này theo tôi hơi khó, vì độ rộng của xung chỉ có từ 0,5 - 1 us có nghĩa là thời gian ngõ vào ở mức "1" rất nhỏ -> lại trở lại vấn đề tìm được các linh kiện rời or IC có thời gian đáp ứng đủ nhỏ là rất khó khăn. Nếu có thể Namqn chỉ giùm tôi luôn
Thanks !
bachelor vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Old 20-07-2008, 11:59 PM   #5
namqn
Trưởng lão PIC bang
 
Tham gia ngày: Feb 2006
Nơi Cư Ngụ: Tp. HCM, Việt Nam
Bài gửi: 3,025
:
Send a message via Yahoo to namqn
Trích:
Nguyên văn bởi bachelor View Post
Càng ngày càng đi chệch ra khỏi các chủ đề của diễn đàn nhưng vẫn theo luồng nên tiện hỏi luôn
1. Namqn có thể chỉ giúp một vài sơ đồ mạch kéo dãn độ rộng xung sử dụng đc trong trường hợp này (với yêu cầu sườn trước của xung sau mạch hiệu chỉnh không được giữ chậm quá nhiều so với xung đầu vào)
2. Cái này theo tôi hơi khó, vì độ rộng của xung chỉ có từ 0,5 - 1 us có nghĩa là thời gian ngõ vào ở mức "1" rất nhỏ -> lại trở lại vấn đề tìm được các linh kiện rời or IC có thời gian đáp ứng đủ nhỏ là rất khó khăn. Nếu có thể Namqn chỉ giùm tôi luôn
Thanks !
1. Đã giới thiệu cho bạn các sơ đồ mạch monostable bên DTVN để bạn nắm được cách thức hoạt động của mạch. Với các xung có độ rộng khoảng 1 us trở lại, bạn có thể dùng các vi mạch chuyên dụng, loại monostable multivibrator, như 74HC123, hay 74HC221. Các vi mạch này chấp nhận xung ngõ vào có độ rộng tối thiểu là 20 ns ở nguồn 5 V, có lẽ thừa đủ cho yêu cầu của bạn.

2. Các vi mạch logic như 74HC32 (4 mạch OR 2 ngõ vào) có thời gian trễ lan truyền khoảng 20 ns ở nguồn 5 V, có lẽ không gặp vấn đề gì với xung của bạn. Không rõ bạn đã thử dùng các vi mạch loại này chưa mà bảo rằng không tìm được các IC có thời gian đáp ứng đủ nhỏ.

Thân,
__________________
Biển học mênh mông, sức người có hạn.

Đang gặp vấn đề cần được giúp đỡ? Hãy dành ra vài phút đọc luồng sau:
http://www.picvietnam.com/forum/showthread.php?t=1263
namqn vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Old 25-07-2008, 11:32 PM   #6
bachelor
Đệ tử 2 túi
 
Tham gia ngày: Jun 2006
Bài gửi: 27
:
Trích:
Nguyên văn bởi namqn View Post
1. Đã giới thiệu cho bạn các sơ đồ mạch monostable bên DTVN để bạn nắm được cách thức hoạt động của mạch. Với các xung có độ rộng khoảng 1 us trở lại, bạn có thể dùng các vi mạch chuyên dụng, loại monostable multivibrator, như 74HC123, hay 74HC221. Các vi mạch này chấp nhận xung ngõ vào có độ rộng tối thiểu là 20 ns ở nguồn 5 V, có lẽ thừa đủ cho yêu cầu của bạn.

2. Các vi mạch logic như 74HC32 (4 mạch OR 2 ngõ vào) có thời gian trễ lan truyền khoảng 20 ns ở nguồn 5 V, có lẽ không gặp vấn đề gì với xung của bạn. Không rõ bạn đã thử dùng các vi mạch loại này chưa mà bảo rằng không tìm được các IC có thời gian đáp ứng đủ nhỏ.

Thân,
Cám ơn vì cả bài bên DTVN
Quả thật chưa thử bao giờ, để xem thị trường ngoài Bắc có con này không đã !
Thanks !
bachelor vẫn chưa có mặt trong diễn đàn   Trả Lời Với Trích Dẫn
Trả lời

Ðiều Chỉnh
Xếp Bài

Quyền Sử Dụng Ở Diễn Ðàn
You may not post new threads
You may not post replies
You may not post attachments
You may not edit your posts

BB code is Mở
Smilies đang Mở
[IMG] đang Mở
HTML đang Tắt

Chuyển đến


Múi giờ GMT. Hiện tại là 03:32 PM.


Được sáng lập bởi Đoàn Hiệp
Powered by vBulletin®
Page copy protected against web site content infringement by Copyscape
Copyright © PIC Vietnam