Sign in to follow this  
Discovery 66

2 И Более Канала Ацп Как Опросить Их (Атмега8)

7 posts in this topic

Как опросить 2 канала ацп котроллера, пример кода для одного канала АЦП как второй канал задействовать, третий и т.п. ? помогите кто чем может, давно уже ищю всю както не конкретно и не понятно, в гугль не отсылать был я там

// Использование АЦП. Цифровой вольтметр
02.
03.#include <avr/io.h>
04.#include <avr/interrupt.h>
05.#include <util/delay.h>
06.
07.//------------------0-----1-----2-----3-----4-----5-----6-----7-----8------9----dp
08.char SEGMENTE[] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, 0x80};
09.
10.volatile unsigned char segcounter = 0;
11.volatile int display = 0;
12.
13.// Прерывание по переполнению T2, динамическая индикация
14.ISR (TIMER2_OVF_vect)
15.{	
16.PORTD = 0xFF;
17.PORTB = (1 << segcounter);
18.
19.switch (segcounter)
20.{	
21.case 0:
22.PORTD = ~(SEGMENTE[display % 10000 / 1000]);
23.break;
24.case 1:
25.PORTD = ~((SEGMENTE[display % 1000 / 100])|0x80); // добавляем десятичную точку
26.break;	
27.case 2:
28.PORTD = ~(SEGMENTE[display % 100 / 10]);
29.break;		
30.}
31.if ((segcounter++) > 2) segcounter = 0;	
32.}
33.
34.volatile unsigned long value;
35.volatile unsigned int adc_counter;
36.
37.// Прерывание по окончанию преобразования АЦП
38.ISR (ADC_vect)
39.{
40.value = value + (ADC*11/4);
41.adc_counter++;
42.}
43.
44.// Главная функция
45.int main (void)
46.{
47.DDRD = 0xFF;
48.DDRB = (1 << PB0)|(1 << PB1)|(1 << PB2)|(1 << PB3);
49.PORTD = 0x00;
50.PORTB = 0x00;
51.DDRC = 0x00;
52.
53.TIMSK |= (1 << TOIE2); // разрешение прерывания по таймеру2
54.TCCR2 |= (1 << CS21); //предделитель на 8
55.
56.ADCSRA = (1 << ADEN) // разрешение АЦП
57.|(1 << ADSC) // запуск преобразования
58.|(1 << ADFR) // непрерывный режим работы АЦП
59.|(1 << ADPS2)|(1 << ADPS1)|(0 << ADPS0) // предделитель на 64 (частота АЦП 125kHz)
60.|(1 << ADIE); // разрешение прерывания
61.
62.ADMUX = (1 << REFS1)|(1 << REFS0) // внутренний ИОН 2,56V
63.|(0 << MUX3)|(0 << MUX2)|(0 << MUX1)|(0 << MUX0); // вход ADC0
64.
65._delay_ms(50);
66.
67.sei(); //глобально разрешаем прерывания
68.
69.while(1)
70.{
71.if (adc_counter > 300) // вычисляем среднее значение АЦП
72.{
73.display = value/adc_counter;
74.adc_counter = 0;
75.value = 0;
76.}	
77._delay_ms(50);
78.}
79.}

Share this post


Link to post
Share on other sites
COKPOWEHEU    273

Объясню на пальцах. Выставляете ADMUX на интересующий канал, запускаете однократное преобразование, дожидаетесь завершения, переставляете на другой, снова запускаете. Запускать АЦП надо после установки ADMUX'a а не до.

Share this post


Link to post
Share on other sites

Старт складской программы по Wi-Fi/ Bluetooth-чипам от Espressif

На склад КОМПЭЛ поступили чипы, модули и отладочные платы от компании Espressif Systems на базе ESP8266 и ESP32. Стоимость всех изделий данной линейки – в 2-3 раза ниже ближайших аналогов, чипы занимают минимальное место на плате, энергоэффективны и универсальны в применении

Подробнее...

hd44780    189

Я пользуюсь такой функцией (опрос, без прерываний):

// Read the AD conversion result
word read_adc(byte adc_input)
{		
word i;

// ADC initialization
// ADC Clock frequency: 1000,000 kHz
// ADC Voltage Reference: AREF pin
// ADC Auto Trigger Source: None
ADMUX=0x00;
ADCSRA=0x83; // 0x10000011 - Enable, CLK/8=1MHz

ADMUX=adc_input;		

// Delay needed for the stabilization of the ADC input voltage
delay_us(50);

// 3 холостых чтения
for (i=0; i<3; i++)
{
// Start the AD conversion (ADSC)
ADCSRA|=0x40;			

// Wait for the AD conversion to complete (ADIF)
while ((ADCSRA & 0x10)==0);

ADCSRA|=0x10;

i=ADCW;	
} // for

// Читать значение АЦП
i=ADCW&0xFFFE;

// ADC off
ADCSRA=0x00;
ADMUX=0x00;
return i;
} // read_adc

Делаются 3 холостых чтения (где-то читал, что так надёжнее). Если не нужно - уберите.

Если по прерываниям, то вместо

while ((ADCSRA & 0x10)==0);

написать родственный цикл с ожиданием флага, который устанавливается в прерывании.

Edited by hd44780

Share this post


Link to post
Share on other sites

помогите у меня сейчас мозг уже взорвется, я не могу понять почему единица записывается в АDMUX |=(1<<MUX0) а нуль нет, после того как там стоит 1...

// Использование АЦП. Цифровой вольтамперметр

#include <avr/io.h>

#include <avr/interrupt.h>

#include <util/delay.h>

//------------------0-----1-----2-----3-----4-----5-----6-----7-----8------9----dp

char SEGMENTE[] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, 0x80};

volatile unsigned char segcounter = 0;

volatile int display_1;

volatile unsigned long value_1;

unsigned int u, g, t = 0, k = 15, z = 4;

volatile unsigned int adc_counter;

// Прерывание по переполнению T2, динамическая индикация

ISR (TIMER2_OVF_vect)

{

PORTD = 0xFF;

PORTB = (1 << segcounter);

switch (segcounter)

{

case 0:

PORTD = ~(SEGMENTE[display_1 % 100000 / 10000]);

break;

case 1:

PORTD = ~((SEGMENTE[display_1 % 10000 / 1000])|0x80); // добавляем десятичную точку

break;

case 2:

PORTD = ~(SEGMENTE[display_1 % 1000 / 100]);

break;

case 3:

PORTD = ~(SEGMENTE[display_1 % 100 / 10]);

break;

}

if ((segcounter++) > 2) segcounter = 0;

}

/***Главная функция***/

int main (void)

{

DDRD = 0xFF;

DDRB = (1 << PB0)|(1 << PB1)|(1 << PB2)|(1 << PB3);

PORTD = 0x00;

PORTB = 0x00;

TIMSK = (1 << TOIE2); // разрешение прерывания по таймеру2

TCCR2 = (1 << CS21); //предделитель на 8

ADCSRA = (1 << ADEN) // разрешение АЦП

|(1 << ADFR) // непрерывный режим работы АЦП

|(1 << ADPS2)|(1 << ADPS1)|(0 << ADPS0); // предделитель на 64 (частота АЦП 125kHz)

ADMUX = (1 << REFS1)|(1 << REFS0)|(0<<MUX3)|(0<<MUX2)|(0<<MUX1)|(0<<MUX0);

_delay_ms(50);

sei(); //глобально разрешаем прерывания

while (1)

{

ADCSRA |= (1 << ADSC); //Начинаем преобразование

while ((ADCSRA&(1 << ADIF))== 0); //Ждем флага окончания преобразования

u = (ADCL|ADCH << 8); // Считываем ADC

value_1 = value_1 + (u * k/z);

adc_counter++;

if (adc_counter > 100)

{

display_1 = (value_1*10/adc_counter);

adc_counter = 0;

value_1 = 0;

_delay_ms(50);

g++;

if (g > 10)

{ADMUX |= (1<<MUX0); k = 5; z = 2;}// переход на АДС1 (выполняется)

if (19 < g)

{ADMUX |= (0<<MUX0); k = 15; z = 4; g = 0;}// вот здесь должен происходить переход на АДС0, но перехода нет

}

}

}

hd44780

я не могу понять предоставленный Вами код, как в этом коде установить канал для чтения? как их менять? с уважением

Share this post


Link to post
Share on other sites

Видео вебинара «Уникальный подход MORNSUN к разработке DC/DC-преобразователей. Что на выходе?»

На сайте КОМПЭЛ доступны материалы вебинара, посвященные последнему поколению DC/DC преобразователей с фиксированным входом R3 от MORNSUN. Вы можете посмотреть видеозапись, ознакомиться с презентацией и ответами на вопросы.

Подробнее...

hd44780    189

Ваша ошибка здесь:

ADMUX = (1 << REFS1)|(1 << REFS0)|(0<<MUX3)|(0<<MUX2)|(0<<MUX1)|(0<<MUX0);

Нуль вы так никогда не запишете. Изучите примитивные лог. операции - http://cxem.net/begi...beginner104.php

Ноль надо писать так:

ADMUX &= ~(1<<MUX0);

или

ADMUX &= ~((1<<MUX0)|(1<<MUX1));

С цветами неохота разбираться :)

Edited by hd44780

Share this post


Link to post
Share on other sites

hd44780

Спасибо, все заработало!!!!

вот полный исправленный код, может таким как я пригодится это для меги 8

// Использование АЦП. Цифровой вольтамперметр

#include <avr/io.h>

#include <avr/interrupt.h>

#include <util/delay.h>

//------------------0-----1-----2-----3-----4-----5-----6-----7-----8------9----dp

char SEGMENTE[] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, 0x80};

volatile unsigned char segcounter = 0;

volatile int display_1;

volatile unsigned long value_1;

unsigned int u, g, t = 0, k = 15, z = 4;

volatile unsigned int adc_counter;

// Прерывание по переполнению T2, динамическая индикация

ISR (TIMER2_OVF_vect)

{

PORTD = 0xFF;

PORTB = (1 << segcounter);

switch (segcounter)

{

case 0:

PORTD = ~(SEGMENTE[display_1 % 100000 / 10000]);

break;

case 1:

PORTD = ~((SEGMENTE[display_1 % 10000 / 1000])|0x80); // добавляем десятичную точку

break;

case 2:

PORTD = ~(SEGMENTE[display_1 % 1000 / 100]);

break;

case 3:

PORTD = ~(SEGMENTE[display_1 % 100 / 10]);

break;

}

if ((segcounter++) > 2) segcounter = 0;

}

/***Главная функция***/

int main (void)

{

DDRD = 0xFF;

DDRB = (1 << PB0)|(1 << PB1)|(1 << PB2)|(1 << PB3);

PORTD = 0x00;

PORTB = 0x00;

TIMSK = (1 << TOIE2); // разрешение прерывания по таймеру2

TCCR2 = (1 << CS21); //предделитель на 8

ADCSRA = (1 << ADEN) // разрешение АЦП

|(1 << ADFR) // непрерывный режим работы АЦП

|(1 << ADPS2)|(1 << ADPS1)|(0 << ADPS0); // предделитель на 64 (частота АЦП 125kHz)

ADMUX = (1 << REFS1)|(1 << REFS0)|(0<<MUX3)|(0<<MUX2)|(0<<MUX1)|(0<<MUX0);

_delay_ms(50);

sei(); //глобально разрешаем прерывания

while (1)

{

ADCSRA |= (1 << ADSC); //Начинаем преобразование

while ((ADCSRA&(1 << ADIF))== 0); //Ждем флага окончания преобразования

u = (ADCL|ADCH << 8); // Считываем ADC

value_1 = value_1 + (u * k/z);

adc_counter++;

if (adc_counter > 300)

{

display_1 = (value_1*10/adc_counter);

adc_counter = 0;

value_1 = 0;

_delay_ms(50);

g++;

if (g > 10)

{ADMUX |= (1<<MUX0); k = 5; z = 2;}// переход на АДС1 (выполняется)

if (19 < g)

{ADMUX &= ~(1<<MUX0); k = 15; z = 4; g = 0;}// переход на АДС0

}

}

}

Share this post


Link to post
Share on other sites

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now

Sign in to follow this  

  • Similar Content

    • By maxssau
      Продам адаптеры USB-I2S. 

       
      Возможности:
       
      стерео ввод/вывод PCM 16-32 бита, 44.1-384 кГц. Ввода DSD нет, как и драйверов для ввода DSD.
      вывод DSD в режиме DoP 64-128, в режиме Native 64-256. Native доступен в Linux без "хитрых" драйверов.
      Тактирование от платы ЦАП/АЦП, частоты 512 fs (22.5792/24.576 МГц). Теоретически возможно и 1024fs(скорости 768кГц и DSD512 Native), но это не опробовано.
      Гальваническая изоляция на Si8662/Si8640.
      Питание возможно как Self так и Bus Powered. В режиме Bus Powered необходимо самостоятельно позаботиться о мастерклоке, т.к. возможны сильные глюки при выключенном генераторе мастерклока.
       
      Тема: 
      срок изготовления 3-5 недель (сильно зависит от поставок процессоров).
      На данный момент полностью реализован интерфейс Legacy. Для отладки Native режима пока нет платы ЦАП, в процессе разработки, будет не раньше осени.
      Цена 6000 + пересылка (в среднем 250 р.).
    • By small girl
      Привет, ребят! Кто возьмется написать на С протокол взаимодействия для сети микроконтроллеров. База: STM32 с bluetooth модулем HC-06,- slave. ПК - master?

      За вознаграждение
    • By Krokodil007
      Продам
      Микроамперметры М42304, 100-0-100, 75шт.
      Микроамперметр М42304, 50-0-50 мкА, 7шт.
      Микроамперметр М42304, 0-50 мкА, 3шт.
      Новые, в упаковках.
      по 120 руб/шт.
       г. Энгельс.
      Возможна отправка почтой или ТК.






    • By LegionKC
      Добрый день.
      Экспериментирую с платой TP4056 и MK ATmega8, конкретнее - пытаюсь собрать простенькое зарядное устройство с выводом параметров при зарядке/разрядке на АЦП МК. Для измерения тока нашел ACS712. Думаю, что получится выводить ток при зарядке (т.е. когда идет питание на TP4056 и нагрузка отключена) и при разрядке (питание выключено, нагрузка подключена). Нагрузка - резистор. Возникла проблема с измерением напряжения на АКБ. Вернее, проблема с отсутствием идей как это сделать. Может какую-нибудь схему делителя напряжения нужно сделать? Прошу помочь советом. Схему из пэинта прилагаю.
      Спасибо.