Перейти к содержанию

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


Discovery 66

Рекомендуемые сообщения

Как опросить 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.}

Ссылка на комментарий
Поделиться на другие сайты

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

Ругался на отсутствие форматирования исходного кода (включая отсутствие осмысленных комментариев и наличие неубранного после конфигуратора мусора) не менее 15 раз.

Часть моих наработок.

Ссылка на комментарий
Поделиться на другие сайты

Сравнительное тестирование аккумуляторов EVE Energy и Samsung типоразмера 18650

Инженеры КОМПЭЛ провели сравнительное тестирование аккумуляторов EVE и Samsung популярного для бытовых и индустриальных применений типоразмера 18650. 

Для теста были выбраны аккумуляторы литий-никельмарганцевой системы: по два образца одного наименования каждого производителя – и протестированы на двух значениях тока разряда: 0,5 А и 2,5 А. Испытания проводились в нормальных условиях на электронной нагрузке EBD-USB от ZKEtech, а зарядка осуществлялась от лабораторного источника питания в режиме CC+CV в соответствии с рекомендациями в даташите на определенную модель. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

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

// 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);

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

Изменено пользователем hd44780
Ссылка на комментарий
Поделиться на другие сайты

Новый аккумулятор EVE серии PLM для GSM-трекеров, работающих в жёстких условиях (до -40°С)

Компания EVE выпустила новый аккумулятор серии PLM, сочетающий в себе высокую безопасность, длительный срок службы, широкий температурный диапазон и высокую токоотдачу даже при отрицательной температуре. 

Эти аккумуляторы поддерживают заряд при температуре от -40/-20°С (сниженным значением тока), безопасны (не воспламеняются и не взрываются) при механическом повреждении (протыкание и сдавливание), устойчивы к вибрации. Они могут применяться как для автотранспорта (трекеры, маячки, сигнализация), так и для промышленных устройств мониторинга, IoT-устройств. Подробнее параметры и результаты тестов новой серии PLM по ссылке.

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

помогите у меня сейчас мозг уже взорвется, я не могу понять почему единица записывается в А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

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

Ссылка на комментарий
Поделиться на другие сайты

Литиевые батарейки и аккумуляторы от мирового лидера  EVE в Компэл

Компания Компэл, официальный дистрибьютор EVE Energy, бренда №1 по производству химических источников тока (ХИТ) в мире, предлагает продукцию EVE как со склада, так и под заказ. Компания EVE широко известна в странах Европы, Америки и Юго-Восточной Азии уже более 20 лет. Недавно EVE была объявлена поставщиком новых аккумуляторных элементов круглого формата для электрических моделей «нового класса» компании BMW.

Продукция EVE предназначена для самого широкого спектра применений – от бытового до промышленного. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

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

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));

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

Изменено пользователем hd44780
Ссылка на комментарий
Поделиться на другие сайты

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

}

}

}

Ссылка на комментарий
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.
Примечание: Ваш пост будет проверен модератором, прежде чем станет видимым.

Гость
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Ответить в этой теме...

×   Вставлено с форматированием.   Восстановить форматирование

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

Загрузка...
  • Последние посетители   0 пользователей онлайн

    • Ни одного зарегистрированного пользователя не просматривает данную страницу
×
×
  • Создать...