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

АЦП 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-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ, также как и для других, очень важен контроль процесса заряда и разряда, а специализированных микросхем для этого вида аккумуляторов не так много. Инженеры КОМПЭЛ подготовили список имеющихся микросхем и возможных решений от разных производителей. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 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 пользователей онлайн

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

    • Чтобы потом не допиливать паровоз до истребителя. Заводские варианты схемотехники куда более предсказуемы. Из "вольностей" в первую очередь как-то решать проблему коммутации, подходящий переключатель для режимов искать история долгая и грустная. Что-то там с реле. На диапазоны 6п4н вроде же ?  пг2 хрен найдёшь, п2г тугое неудобство(  Рассыпушный ацп делать, боже упаси. Ну разве что был грешен, побаловался когда-то пнч-ацп. С 1108пп1 и 155 серией
    • Но зачем повторять 1в1? Там дана просто топология   Но не совсем же с нуля. Маловероятно, что вы найдете расширение диапазона 7135 до 200мВ по тому же методу, что это реализовано в В7-38, тем не менее, это вполне реализуемо. А вот до 20мВ я спустить диапазон не смог Что-то предусилитель интегратора отказывается вести себя хорошо   Для сравнения, та же точка (вход интегратора) при 200мв пределе, усиление х10   Ну и 2В Госпаде боже лтц2400, на этой штуке можно и 6 разрядник собрать, лютый АЦП, какие 4.5 разряда Цена на него сейчас конечно негуманная кстати.
    • Что то не нашёл примеров такой реализации замены. С нуля целесообразность подобного действа под вопросом. Тут мои полномочия всё) Как бы почему смотрю на hm8011-3, возможность реализации +/- норм настольного мультиметра без всякой там жести в виде кодинга и микро-мелких современных элементов. Для калибровки валяется где-то в1-12 и прецизионных резисторов если моих не хватит, есть где заказать.  На данный момент разве что качественного файла нет, пытался там всякими улучшайками чего сделать, увы. Пойду через впн лазить по уголочкам всяким, вдруг всплывёт чего. ------------- 8012, 80c32 + 27c512. Чертежи плат есть, осталось hex раздобыть 
    • Вот на этом фото? Я думал  тут радиатор будет под ними, и они к нему "пузом" будут прикручиваться, а дырочки - для отвёртки. У @asng60 транзисторы в наружную сторону "мордой" смотрят. "Пузом" на плату ложатся. Плата так разведена. Вот и непонятно, как их "пузом" к радиатору прикручивать.
    • Что касаемо модели Парафина, я бы порекомендовал товарищу Джексону проверить ток покоя выходного каскада, это первым делом, и вторым делом- проверить модель ОУ. Ибо в ОМ нет никаких оу а в парафине- есть. 
    • сломался Ну точно сломался! Все, сломался, однозначно! Несите в мастерскую!
×
×
  • Создать...