Thông báo

Collapse
No announcement yet.

TẠi sao khi sỬ dỤng 2 kÊnh adc trong atmega32

Collapse
X
 
  • Lọc
  • Giờ
  • Show
Clear All
new posts

  • #31
    ?????????????????????????????quả là cao thu dau mung mu
    thay vì phải /1000,tốn dung lượng, nên /10 thui.nên tự chuyển số ->mã ký tự......
    hãy quý những j hiện tại mình đang có

    Comment


    • #32
      hay quá, thank các bác

      Comment


      • #33
        Nguyên văn bởi ptnksmart Xem bài viết
        Chả hiểu tại sao khi sử dụng cùng 2 kênh ADC trong ATMEGA32 thì chỉ có 1 kênh hoạt động kênh còn lại chạy theo kênh kia.

        Cụ thể là dùng 2 loại cảm biến ánh sáng và nhiệt độ đưa vào 2 kênh sau đó phát ra LCD thì nó hiện kết quả chỉ đúng cho 1 kênh, kênh còn lại thì lại hiện kết quả phụ thuộc. Ví dụ như thông số nhiệt độ là 32 (ở kênh ADC1), ánh sáng là 50 (ở kênh ADC0) thì khi hiện lên LCD nó hiện là 32 32 thay đổi thông số nhiệt độ thì thông số ánh sáng trên LCD chạy theo còn thay đổi thông số ánh sáng thì thông số trên LCD vẫn nằm yên.Chả hiểu làm sao.

        Nhưng khi tách ra làm từng kênh thì nó lại chạy đúng.

        Mình dùng winavr và protues mô phỏng.
        Mong bà con cô bác giúp hộ.


        code mình viết:

        #include <avr/io.h>
        #include <util/delay.h>
        #define AREF_MODE 0
        #define AVCC_MODE (1<<REFS0)
        #define INT_MODE (1<<REFS1)|(1<<REFS0)
        #define ADC_VREF_TYPE AREF_MODE
        uint16_t read_adc(unsigned char adc_channel);
        uint16_t nhietdo,dosang,A[15];
        int main(){
        ADCSRA=(1<<ADEN)|(1<<ADPS2)|(1<<ADPS1);
        ADMUX=ADC_VREF_TYPE;
        //khoi dong chuyen doi ADC
        init_LCD();
        clr_LCD();
        while(1){
        dosang=read_adc(0)/2;
        lcd_putnum(dosang,2,9);
        _delay_us(1000);
        nhietdo=(read_adc(1)-2)/2;
        lcd_putnum(nhietdo,2,13);
        _delay_us(1000);
        }
        }
        uint16_t read_adc(unsigned char adc_channel) {
        ADMUX|=adc_channel;
        ADCSRA|=(1<<ADSC);
        //wait for conversion complete
        loop_until_bit_is_set(ADCSRA,ADIF);
        //while (!(ADCSRA & (1<<ADIF)));
        return ADCW;
        }
        dùng tụ điện 100n đấu vào chân adc cần đọc xuống mass. thì hết bị sai số.

        Comment


        • #34
          ADMUX|=adc_channel;
          delay_ms(1);
          - Còn code của mình đây:
          void Channel_Select(unsigned char ch)
          {
          ADMUX=ch|ADC_VREF_TYPE;
          _delay_us(10); // Phải có delay này, nếu không sẽ bị trùng kênh
          }

          Comment


          • #35
            Mình cũng dùng 2 kênh ADC0, ADC1 chuyển đổi lần lượt và gán kết quả tương ứng vào các biến d, e. Nhưng cuối cùng ra kết quả d=e (d sai, e đúng). Các kiểm tra đã thực hiện:
            1. Đặt lệnh delay_ms(1); giữa hai lần chuyển đổi: kết quả vẫn là d=e (d sai, e đúng).
            2. Thử đặt while(1); dừng chương trình tại chỗ sau chuyển đổi ADC0 thì nhận được d đúng.

            Đây là đoạn chương trình trong CodeVisionAVR:

            ADCSRA|=(1<<ADSC); //Khoi dong ADC0
            while(!(ADCSRA & (1<<ADIF)));//cho den khi bit (ADCSRA,ADIF) duoc set
            ADCSRA |= (1 << ADIF); // Xóa cờ ADIF
            d=ADCW; //ket qua ADC0

            delay_ms(1);


            ADMUX |= (1<<MUX0); // Chon ADC1
            ADCSRA|=(1<<ADSC); //Khoi dong ADC1
            while(!(ADCSRA & (1<<ADIF))); //cho den khi bit (ADCSRA,ADIF) duoc set
            ADCSRA |= (1 << ADIF); // Xóa co ADIF
            e=ADCW; //ket qua ADC1

            Mong được sự góp ý của ACE cộng đồng để khắc phục lỗi, mình trân trọng cảm ơn!!!

            Comment


            • #36
              Nguyên văn bởi tuecamphu1 Xem bài viết
              Mình cũng dùng 2 kênh ADC0, ADC1 chuyển đổi lần lượt và gán kết quả tương ứng vào các biến d, e. Nhưng cuối cùng ra kết quả d=e (d sai, e đúng). Các kiểm tra đã thực hiện:
              1. Đặt lệnh delay_ms(1); giữa hai lần chuyển đổi: kết quả vẫn là d=e (d sai, e đúng).
              2. Thử đặt while(1); dừng chương trình tại chỗ sau chuyển đổi ADC0 thì nhận được d đúng.

              Đây là đoạn chương trình trong CodeVisionAVR:

              ADCSRA|=(1<<ADSC); //Khoi dong ADC0
              while(!(ADCSRA & (1<<ADIF)));//cho den khi bit (ADCSRA,ADIF) duoc set
              ADCSRA |= (1 << ADIF); // Xóa cờ ADIF
              d=ADCW; //ket qua ADC0

              delay_ms(1);


              ADMUX |= (1<<MUX0); // Chon ADC1
              ADCSRA|=(1<<ADSC); //Khoi dong ADC1
              while(!(ADCSRA & (1<<ADIF))); //cho den khi bit (ADCSRA,ADIF) duoc set
              ADCSRA |= (1 << ADIF); // Xóa co ADIF
              e=ADCW; //ket qua ADC1

              Mong được sự góp ý của ACE cộng đồng để khắc phục lỗi, mình trân trọng cảm ơn!!!
              Cái hàm delay đặt chỗ đó không hợp lý.
              Thường là đặt sau khi chuyển kênh.
              Đêm nay tớ không ngủ - ngày mai tớ ngủ bù

              Comment


              • #37
                Đúng vậy, vị trí delay không phù hợp. Tuy nhiên vấn đề không ở chỗ đó.
                Mình đã sửa lại lệnh chọn kênh ADC1 thành ADMUX = (ADMUX & 0xF0) | 0x01; (giữ nguyên 4 bit dầu thanh ghi ADMUX, xóa 4 bit cuối MUX3:0, đặt lại MUX3:0=0001) và vấn đề đã được giải quyết.

                Comment

                Về tác giả

                Collapse

                ptnksmart Tìm hiểu thêm về ptnksmart

                Bài viết mới nhất

                Collapse

                Đang tải...
                X