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

АЦП Atmega8


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

Здравствуйте, вопрос по ацп atmega8, задача примитивная - сделать вольтметр, подскажите, пожалуйста, как вывести данные из ADCW, и нужно ли делать прерывание по таймеру или можно запустить нерперывное преобразование?если непрерывное, то как считывать данные?

с остальными регистрами вроде разобрался по даташиту, прошу строго не судить - только начинаю разбираться...

 

TEST_DISP_PROBA_2.atsln

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

Реклама: ООО ТД Промэлектроника, ИНН: 6659197470, Тел: 8 (800) 1000-321

unsigned int adc_data;
................................
// прерывание по завершению преобразования АЦП
interrupt [ADC_INT] void adc_isr(void)
{adc_data=ADCW;
 ADCSRA.ADSC=1;  //повторный запуск преобразования АЦП.
 //можно перезапускать здесь, можно в прерывании таймера
}
.................................
void main(void)
{................................
 //инициализация АЦП
 ADMUX=....;
 ADCSRA=....;
 ADCSRA.ADSC=1;  //первый запуск преобразования АЦП.
 //можно запускать здесь, можно в прерывании таймера
.................................
};

Код под CVAVR

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

20% скидка на весь каталог электронных компонентов в ТМ Электроникс!

Акция "Лето ближе - цены ниже", успей сделать выгодные покупки!

Плюс весь апрель действует скидка 10% по промокоду APREL24 + 15% кэшбэк и бесплатная доставка!

Перейти на страницу акции

Реклама: ООО ТМ ЭЛЕКТРОНИКС, ИНН: 7806548420, info@tmelectronics.ru, +7(812)4094849

спасибо за ответ. Насколько я понял из даташита, преобразованное значение хранится в  ADCH:ADCL и битом ADLAR выставляем выравнивание, если мы берем, например, переменную v (напряжение) и записываем, что v=ADCH, предварительно выставив бит ADLAR, то ничего не работает...подскажите, пожалуйста, как правильно выводить (считывать) преобразованное значение?

Изменено пользователем СКУПОЙ
II.9
Ссылка на комментарий
Поделиться на другие сайты

Выбираем схему BMS для заряда литий-железофосфатных (LiFePO4) аккумуляторов

Обязательным условием долгой и стабильной работы Li-FePO4-аккумуляторов, в том числе и производства EVE Energy, является применение специализированных BMS-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ очень важен контроль процесса заряда и разряда для избегания воздействия внешнего зарядного напряжения после достижения 100% заряда. Инженеры КОМПЭЛ подготовили список таких решений от разных производителей. Подробнее>>

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

Здравствуйте, подскажите, пожалуйста, где ошибка в программе, просто текст выводится, а вот с выводом данных из АЦП проблема - мигающий курсор вместо значения..

 

float v;
    InitLcd();
    ADMUX=0b01000101; // канал 5
    _delay_us(10); // стабилизация входа...
    ADCSRA|=0b01000000; // Старт преобразования.
    while ((ADCSRA & 0x10)==0); // ждём ADIF - флаг завершении преобразования
    ADCSRA|=0x10; // сброс ADIF - флаг завершении преобразования
    v=ADCW*0.00489; // (напряжение) (00000011 11111111).
    char buffer [5];
    
    //InitADC();
    sprintf(buffer, "U=%.1f", v);
    LcdWriteCom(0b00000001);// команда очистки экрана
    _delay_ms(100);
    Send_Str(buffer,5); // Отправляем строку
    _delay_ms(100); //Пауза
    ADCSRA |=(1<<ADSC); //Запускаем следующее преобразование
    //ADCSRA |=(1<<ADSC);
    while(1);
    {
    }

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

21 час назад, a_slepnev сказал:

битом ADLAR выставляем выравнивание

При ADLAR=0 АЦП выполняет 10-битное преобразование. Сначала считываются 8 младших бит из ADCL, затем 2 старших из ADCH. Конечный результат ADCL+ADCH*0xFF Если ADLAR=1, то АЦП выполняет 8-битное преобразование. Весь результат сохраняется в ADCH.

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

3 минуты назад, Sergey-Ufa сказал:

Весь результат сохраняется в ADCH.

А тогда ADCW для чего? и еще такая проблема, мне нужно вывести значение ацп на wh0802, если, например, я пишу lcd_data(ADCW) - на экране какой-то непонятный символ, lcd_data- запись данных, а если:

void Send_Str (char* str,char length)// Отправить строку LCD, длины length
{
    for (int i = 0;i<length;i++)
    {
        LcdWriteData(*str);
        str++;
        _delay_us(60);
    }
}

float v;
InitLcd();
ADMUX=0b01000101; // канал 5
_delay_us(10); // стабилизация входа...
ADCSRA|=0b01000000; // Старт преобразования.
while ((ADCSRA & 0x10)==0); // ждём ADIF - флаг завершении преобразования
ADCSRA|=0x10; // сброс ADIF - флаг завершении преобразования
v=ADCW*0.00489; // (напряжение) 
char buffer [10];
sprintf(buffer, "U=%.1f", v); 
LcdWriteCom(0b00000001);
_delay_ms(10);
Send_Str(buffer,10); // Отправляем строку
_delay_ms(100); //Пауза
ADCSRA |=(1<<ADSC); //Запускаем следующее преобразование

while(1);
{
}

 

то на дисплее просто мигающий курсор, помогите, пожалуйста, разобраться, где ошибка

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

4 часа назад, a_slepnev сказал:

ждём ADIF - флаг завершении преобразования

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

Можно проверять другой флаг ADSC, он работает без прерываний.

while ((ADCSRA & 0x40)==1); // ждём ADSC = 0 - сброс флага при завершении преобразования

 

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

ADCW это представление регистров данных АЦП в виде одного 16-битного слова в некоторых Си-компиляторах для AVR. Он имеется в CVAVR. Со Студией я не работаю, поэтому не могу сказать есть он там или нет. Компилятор сам формирует  код для чтения данных АЦП за одну операцию

unsigned int Data=ADCW;

Если у вас ADLAR=1, то использование ADCW смысла не имеет и дает результат помноженный на 255. Здесь достаточно прочитать только ADCH

unsigned char Data=ADCH;

1 час назад, a_slepnev сказал:

если, например, я пишу lcd_data(ADCW) - на экране какой-то непонятный символ

Правильно.

1) Вы должны отправлять в LCD строковое представление числа, а не само число.

2) lcd_data требует unsigned char, а ADCW это фактически unsigned int

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

1 час назад, Sergey-Ufa сказал:

2) lcd_data требует unsigned char, а ADCW это фактически unsigned int

Не могли бы подсказать, где ошибка? на дисплее просто мигающий курсор

#include <avr/io.h>
//#include <intrinsics.h>
#define F_CPU 8000000
#include <util/delay.h>
#include <avr/interrupt.h>
#include <stdio.h>
// #define _delay_us(us)     __delay_cycles((F_CPU / 1000000) * (us));
// #define _delay_ms(ms)     __delay_cycles((F_CPU / 1000) * (ms));

//порт к которому подключена шина данных ЖКД
#define PORT_DATA PORTD
#define PIN_DATA  PIND
#define DDRX_DATA DDRD

//порт к которому подключены управляющие выводы
#define PORT_SIG PORTB
#define PIN_SIG  PINB
#define DDRX_SIG DDRB

//номера выводов микроконтроллера
//к которым подключены управляющие выводы ЖКД
#define RS 4
#define RW 5
#define EN 6

//макросы для работы с битами
#define ClearBit(reg, bit)       reg &= (~(1<<(bit)))
#define SetBit(reg, bit)          reg |= (1<<(bit))

 

//функция записи команды
void LcdWriteCom(unsigned char data)
{
    ClearBit(PORT_SIG, RS);    // устанавливаем RS в 0
    PORT_DATA = data;        // выводим данные на шину
    SetBit(PORT_SIG, EN);    // устанавливаем Е в 1
    _delay_ms(60);
    ClearBit(PORT_SIG, EN);    // устанавливаем Е в 0
    _delay_ms(60);
}

//функция записи данных
void LcdWriteData(unsigned char data)
{
    SetBit(PORT_SIG, RS);     //устанавливаем RS в 1
    PORT_DATA = data;        //выводим данные на шину
    SetBit(PORT_SIG, EN);     //устанавливаем Е в 1
    _delay_ms(60);
    ClearBit(PORT_SIG, EN);    // устанавливаем Е в 0
    _delay_ms(60);
}

//функция инициализации
void InitLcd(void)
{
    //настраиваем порты ввода/вывода
    DDRX_DATA = 0xff;
    PORT_DATA = 0xff;
    DDRX_SIG = 0xff;
    PORT_SIG |= (1<<RW)|(1<<RS)|(1<<EN);
    ClearBit(PORT_SIG, RW);
    _delay_ms(60);
    LcdWriteCom(0x38); //0b00111000 - 8 разрядная шина, 2 строки
    LcdWriteCom(0x0f);  //0b00001111 - дисплей, курсор, мерцание включены
    LcdWriteCom(0x01);  //0b00000001 - очистка дисплея
    _delay_ms(60);
    LcdWriteCom(0x06);  //0b00000110 - курсор движется вправо, сдвига нет
    
}
void InitADC(void)
{
ADMUX |=(1<<REFS0)|(1<<ADLAR);//AVCC — источник опорного напряжения
ADCSRA |=(1<<ADEN)|(1<<ADIE)|(1<<ADPS1)|(1<<ADPS2);
}
void Send_Str (char* str,char length)// Отправить строку LCD, длины length
{
    for (int i = 0;i<length;i++)
    {
        LcdWriteData(*str);
        str++;
        _delay_us(60);
    }
}

ISR (ADC_vect) //Прерывание окончания преобразования АЦП
{
    float Voltage = ADCH*0.00489;//Результат преобразования хранится в ADCW = ADCH:ADCL
    char buffer [5];// Строка для отправки LCD
    sprintf(buffer,"U=%.2f V",Voltage);// Преобразуем значение напряжения к строке
    LcdWriteCom(0b00000001);// Подаем команду очистки экрана
    _delay_ms(2);// Пауза для ожидания завершения исполнения команды
    Send_Str(buffer,10); // Отправляем строку
    _delay_ms(100); //Пауза
    ADCSRA |=(1<<ADSC); //Запускаем следующее преобразование
}

int main(void)
{
    InitADC();
    InitLcd();
    while(1)
    {
        
    }
}

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

 

(ADCSRA & 0x40)==1

Это никогда не будет истинным!

Если забанить всех, кто набрался смелости думать независимо, здорово будет на форуме - как на кладбище: тишина, птички поют...

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

5 часов назад, Sergey-Ufa сказал:

2) lcd_data требует unsigned char, а ADCW это фактически unsigned int

Не могли бы подсказать, где ошибка? на дисплее просто мигающий курсор

Скрытый текст



#include <avr/io.h>
//#include <intrinsics.h>
#define F_CPU 8000000
#include <util/delay.h>
#include <avr/interrupt.h>
#include <stdio.h>
// #define _delay_us(us)     __delay_cycles((F_CPU / 1000000) * (us));
// #define _delay_ms(ms)     __delay_cycles((F_CPU / 1000) * (ms));

//порт к которому подключена шина данных ЖКД
#define PORT_DATA PORTD
#define PIN_DATA  PIND
#define DDRX_DATA DDRD

//порт к которому подключены управляющие выводы
#define PORT_SIG PORTB
#define PIN_SIG  PINB
#define DDRX_SIG DDRB

//номера выводов микроконтроллера
//к которым подключены управляющие выводы ЖКД
#define RS 4
#define RW 5
#define EN 6

//макросы для работы с битами
#define ClearBit(reg, bit)       reg &= (~(1<<(bit)))
#define SetBit(reg, bit)          reg |= (1<<(bit))


//функция записи команды
void LcdWriteCom(unsigned char data)
{
    ClearBit(PORT_SIG, RS);    // устанавливаем RS в 0
    PORT_DATA = data;        // выводим данные на шину
    SetBit(PORT_SIG, EN);    // устанавливаем Е в 1
    _delay_ms(60);
    ClearBit(PORT_SIG, EN);    // устанавливаем Е в 0
    _delay_ms(60);
}

//функция записи данных
void LcdWriteData(unsigned char data)
{
    SetBit(PORT_SIG, RS);     //устанавливаем RS в 1
    PORT_DATA = data;        //выводим данные на шину
    SetBit(PORT_SIG, EN);     //устанавливаем Е в 1
    _delay_ms(60);
    ClearBit(PORT_SIG, EN);    // устанавливаем Е в 0
    _delay_ms(60);
}

//функция инициализации
void InitLcd(void)
{
    //настраиваем порты ввода/вывода
    DDRX_DATA = 0xff;
    PORT_DATA = 0xff;
    DDRX_SIG = 0xff;
    PORT_SIG |= (1<<RW)|(1<<RS)|(1<<EN);
    ClearBit(PORT_SIG, RW);
    _delay_ms(60);
    LcdWriteCom(0x38); //0b00111000 - 8 разрядная шина, 2 строки
    LcdWriteCom(0x0f);  //0b00001111 - дисплей, курсор, мерцание включены
    LcdWriteCom(0x01);  //0b00000001 - очистка дисплея
    _delay_ms(60);
    LcdWriteCom(0x06);  //0b00000110 - курсор движется вправо, сдвига нет
    
}
void InitADC(void)
{
ADMUX |=(1<<REFS0)|(1<<ADLAR);//AVCC — источник опорного напряжения
ADCSRA |=(1<<ADEN)|(1<<ADIE)|(1<<ADPS1)|(1<<ADPS2);
}
void Send_Str (char* str,char length)// Отправить строку LCD, длины length
{
    for (int i = 0;i<length;i++)
    {
        LcdWriteData(*str);
        str++;
        _delay_us(60);
    }
}

ISR (ADC_vect) //Прерывание окончания преобразования АЦП
{
    float Voltage = ADCH*0.00489;//Результат преобразования хранится в ADCW = ADCH:ADCL
    char buffer [5];// Строка для отправки LCD
    sprintf(buffer,"U=%.2f V",Voltage);// Преобразуем значение напряжения к строке
    LcdWriteCom(0b00000001);// Подаем команду очистки экрана
    _delay_ms(2);// Пауза для ожидания завершения исполнения команды
    Send_Str(buffer,10); // Отправляем строку
    _delay_ms(100); //Пауза
    ADCSRA |=(1<<ADSC); //Запускаем следующее преобразование
}

int main(void)
{
    InitADC();
    InitLcd();
    while(1)
    {
        
    }
}


 

 

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

16 часов назад, ARV сказал:

Это никогда не будет истинным!

Да, это мой любимый баг в программах :D. Бывает, когда бездумно меняешь 0 на 1 для инверсии условия.

Надо так, конечно

while ((ADCSRA & 0x40) != 0); // ждём ADSC = 0 - сброс флага при завершении преобразования


 

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

1 час назад, Yurkin2015 сказал:

Надо так, конечно

Надо придерживаться правила "убирай лишнее - будет правильно". А еще есть правило "магические числа открывают ворота в ад". 

while(ADCSRA & (1<<ADSC));
// а еще лучше
while(bit_is_set(ADCSRA, ADSC));

 

Если забанить всех, кто набрался смелости думать независимо, здорово будет на форуме - как на кладбище: тишина, птички поют...

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

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

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

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

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

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

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

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

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

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

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