В данной статье мы рассмотрим схему для измерения температуры, построенную на основе микроконтроллера ATmega32 (семейство AVR) и датчика LM35. LM35 представляет собой датчик линейного напряжения. Как известно, температура обычно измеряется в градусах Цельсия или фаренгейтах. Выходная шкала сенсора LM35 отградуирована в градусах Цельсия.
Принцип работы датчика LM35
Подобно транзистору датчик LM35 имеет три контакта: VCC (Voltage constant current - напряжение постоянного тока), GND (земля) and OUTPUT (выход). Датчик обеспечивает переменное напряжение на выходе, зависящее от измеряемой температуры.
Как показано на представленном рисунке, на каждый дополнительный градус Цельсия датчик обеспечивает увеличение напряжения на 10 мВ на своем выходе. Таким образом, если температура составляет 0 градусов Цельсия, то на выходе датчика будет 0 В, для 10 градусов Цельсия на выходе датчика будет +100 мВ, если температура 25 градусов Цельсия, то на выходе датчика будет +250 мВ.
Теперь мы можем измерять значение температуры с помощью анализа напряжения на выходе датчика LM35. Сделать мы это сможем с помощью цифро-аналогового преобразователя микроконтроллера ATmega32A. Полученное после конверсии (преобразования из аналога в цифру) цифровое значение показывается на жидкокристаллическом (ЖК) дисплее 16x2 – это и будет значение температуры.
Необходимые компоненты
Аппаратное обеспечение
- Микроконтроллер ATmega32 (купить на AliExpress).
- Программатор AVR-ISP (купить на AliExpress), USBASP (купить на AliExpress) или другой подобный.
- ЖК дисплей JHD_162ALCD (16x2, 16 символов, 2 строки) (купить на AliExpress).
- Датчик температуры LM35 (купить на AliExpress).
- Конденсатор 100 мкФ (купить на AliExpress).
- Конденсатор 100 нФ (купить на AliExpress).
- Источник питания с напряжением 5 Вольт.
Реклама: ООО "АЛИБАБА.КОМ (РУ)" ИНН: 7703380158
Программное обеспечение
- Atmel Studio версии 6.1 (или выше).
- Progisp или flash magic (необязательно).
Работа схемы
Схема устройства приведена на следующем рисунке.
В представленной схеме PORTB микроконтроллера ATmega32 соединен с портом данным жидкокристаллического (ЖК) дисплея. При этом следует помнить о том, что необходимо деактивировать JTAG интерфейс микроконтроллера на порту PORTC при помощи изменения фьюзов (fuse bytes) чтобы использовать PORTC как обычный порт ввода/вывода. В ЖК дисплее (если мы не хотим использовать черный цвет) можно задействовать только 14 его контактов: 8 контактов для передачи данных (7-14 или D0-D7), 2 контакта для подачи питания (1&2 или VSS&VDD или gnd&+5v), 3-й контакт для управления контарстностью, 3 контакта для управления (RS&RW&E).
В представленной схеме мы использовали только 2 контакта управления ЖК дисплея для лучшего понимания работы схемы. Бит контраста и READ/WRITE используются нечасто, поэтому они могут быть замкнуты на землю. Это обеспечивает ЖК дисплею максимальную контрастность и переводит его в режим чтения. Теперь нам всего лишь нужно контролировать контакты ENABLE и RS чтобы передавать на ЖК дисплей символы и данные. Также на нашем сайте вы можете прочитать более подробную статью о подключении ЖК дисплея к микроконтроллеру AVR ATmega32.
В схеме необходимо сделать следующие соединения с ЖК дисплеем:
PIN1 или VSS - земля
PIN2 или VDD или VCC - +5v питание
PIN3 или VEE - земля (обеспечивает максимальный контраст ЖК дисплею)
PIN4 или RS (Register Selection) – контакт PD6 микроконтроллера
PIN5 или RW (Read/Write) - земля (переводит ЖК дисплей в режим чтения что упрощает взаимодействие с ним для начинающих)
PIN6 или E (Enable) - контакт PD5 микроконтроллера
PIN7 или D0 - контакт PB0 микроконтроллера
PIN8 или D1 - контакт PB1 микроконтроллера
PIN9 или D2 - контакт PA2 микроконтроллера
PIN10 или D3 - контакт PB2 микроконтроллера
PIN11 или D4 - контакт PB4 микроконтроллера
PIN12 или D5 - контакт PB5 микроконтроллера
PIN13 или D6 - контакт PB6 микроконтроллера
PIN14 или D7 - контакт PB7 микроконтроллера
В схеме мы использовали 8-битную связь (D0-D7) ЖК дисплея с микроконтроллером, хотя можно было ограничиться и 4-битной – но в этом случае код программы стал бы немного сложнее. Таким образом, мы использовали 10 контактов ЖК дисплея, 8 из которых будут использоваться для передачи данных и 2 для управления.
Следует отметить, что напряжение на выходе сенсора LM35 не совсем линейное, оно немного зашумлено. Для устранения этого шума используется конденсатор, подключенный к выходам сенсора.
Перед тем как пойти дальше остановимся немного на принципах работы аналого-цифрового преобразователя (АЦП) микроконтроллера ATmega32A. В данном микроконтроллере мы можем использовать АЦП на любом из восьми каналов PORTA, не важно какой мы выберем – разницы нет. В данной схеме мы использовали канал 0 (контакт PIN0) PORTA. В ATmega32A АЦП имеет разрешение (разрешающую способность) 10 бит, таким образом микроконтроллер способен реализовать чувствительность равную Vref/2^10, то есть если опорное напряжение (Vref) равно 5В мы получим цифровой инкремент на выходе 5/2^10 = 5мВ. Таким образом, на каждое приращение напряжения на 5мВ мы будем получать один дополнительный инкремент цифрового выхода АЦП.
Для обеспечения работы схемы мы должны установить значения регистров АЦП следующим образом:
- Сначала мы должны активировать АЦП микроконтроллера.
- Поскольку мы собираемся измерять только комнатные температуры нам не нужны нижние значения напряжения (ниже значения 1000 мВ на выходе сенсора LM35). Поэтому мы можем установить максимальное значение опорного напряжения АЦП равное 2.5В.
- АЦП микроконтроллера в нашей схеме будет начинать действовать при внешнем воздействии (не от действий пользователя), поэтому нам следует установить его в режим непрерывного преобразования (free running mode): в этом режиме запуск преобразований выполняется непрерывно через определенные интервалы времени.
- В любом АЦП частота преобразования аналогового значения в цифровое и точность цифрового выхода обратно пропорциональны. То есть для лучшей точности цифрового выхода мы должны выбрать меньшую частоту. Для этого мы должны установить коэффициент деления предделителя АЦП в максимальное значение (128). Поскольку мы используем внутреннюю частоту микроконтроллера 1 МГц, то значение частоты преобразования АЦП будет равно 1000000/128.
Теперь нам нужно установить правильные значения в регистрах АЦП.
RED (красный, ADEN): этот бит устанавливается чтобы задействовать функции АЦП в ATmega32A.
BLUE (синий, REFS1, REFS0): эти два бита используются для установки опорного напряжения (максимального входного напряжения, которое мы собираемся обрабатывать). Поскольку мы будем использовать опорное напряжение равное 2.56V, биты REFS0 и REFS1 необходимо выставить в соответствии с приведенной таблицей.
LIGHT GREEN (светло зеленый, ADATE): этот бит должен быть установлен чтобы АЦП работал непрерывно (в режиме непрерывного преобразования).
PINK (розовый, MUX0-MUX4): эти 5 бит используются чтобы задать входной канал. Поскольку мы будем использовать ADC0 (PIN0) то, как следует из ниже приведенной таблицы, нам нет необходимости устанавливать все эти биты.
BROWN (коричневый, ADPS0-ADPS2): эти три бита используются для установки коэффициент деления предделителя АЦП. Поскольку мы используем коэффициент деления предделителя 128, мы должны установить все эти три бита.
DARK GREEN (темно-зеленый, ADSC): этот бит необходимо установить для того чтобы АЦП начал осуществлять преобразование. Далее в программе мы можем его сбросить (в 0) если нам нужно будет остановить процесс аналого-цифрового преобразования.
Объяснение работы программы
Рассмотрим текст программы на языке С (Си) с комментариями.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
#include <avr/io.h> // заголовок чтобы разрешить контроль данных на контактах #define F_CPU 1000000 // задание тактовой частоты #include <util/delay.h> // заголовок чтобы задействовать функции задержки в программе #define E 5 // задействуем 5-й контакт PORTD (“enable”), поскольку он соединен с контактом “enable” ЖК дисплея #define RS 6 // задействуем выбор регистра (“registerselection”) на 6-м контакте PORTD, поскольку он соединен с контактом RS ЖК дисплея void send_a_command(unsigned char command); void send_a_character(unsigned char character); void send_a_string(char *string_of_characters); int main(void) { DDRB = 0xFF; // установка portB и portD на вывод данных DDRD = 0xFF; _delay_ms(50);// задержка 50ms DDRA = 0;// установка portA на ввод данных ADMUX |=(1<<REFS0)|(1<<REFS1);// установка опорного напряжения для АЦП ADCSRA |=(1<<ADEN)|(1<<ADATE)|(1<<ADPS0)|(1<<ADPS1)|(1<<ADPS2); // активация АЦП, установка свободного режима, установка шкалы АЦП 128 int16_t COUNTA = 0;// хранение значения цифрового выхода char SHOWA [3];// отображение цифрового выхода в качестве температуры на 16*2 ЖК дисплее send_a_command(0x01); // очистить экран 0x01 = 00000001 _delay_ms(50); send_a_command(0x38);// сообщаем ЖК дисплею что мы будем использовать 8 битный режим команд/данных _delay_ms(50); send_a_command(0b00001111);// включаем экран и мигание курсора на ЖК дисплее ADCSRA |=(1<<ADSC);// старт АЦП while(1) { COUNTA = ADC/4; // поскольку мы используем разрешение (2.56/2^10 = 0.0025) 2.5 мВ у нас получится инкремент 4 на каждые дополнительные 10 мВ на входе, что означает что на каждый дополнительный градус температуры мы получим инкремент 4 в цифровом значении. Таким образом чтобы получить значение температуры мы должны разделить значение на выходе АЦП на 4 send_a_string ("CIRCUIT DIGEST ");// отображение имени send_a_command(0x80 + 0x40 + 0); // установка курсора на 1-ю позицию 2-й строки send_a_string ("Temp(C)=");// отображение имени send_a_command(0x80 + 0x40 + 8); // установка курсора на 9-ю позицию 2-й строки itoa(COUNTA,SHOWA,10); //command for putting variable number in LCD(variable number, in which character to replace, which base is variable(ten here as we are counting number in base10)) send_a_string(SHOWA); // сообщаем дисплею чтобы он показал символ (замененный номером переменной) после позиционирования курсора на ЖК дисплее send_a_string (" "); send_a_command(0x80 + 0);// возвращаемся на первую позицию первой строки } } void send_a_command(unsigned char command) { PORTA = command; PORTD &= ~ (1<<RS); // устанавливаем 0 в RS чтобы сообщить дисплею что мы будем передавать команду PORTD |= 1<<E; // сообщаем ЖК дисплею чтобы он принял команду/данные _delay_ms(50); PORTD &= ~1<<E;// сообщаем ЖК дисплею что мы закончили передачу данных PORTA= 0; } void send_a_character(unsigned char character) { PORTA= character; PORTD |= 1<<RS;// сообщаем ЖК дисплею что мы будем передавать данные (не команду) PORTD |= 1<<E;// сообщаем ЖК дисплею чтобы он начал прием команды/данных _delay_ms(50); PORTD &= ~1<<E;// сообщаем ЖК дисплею что мы закончили передачу данных/команды PORTA = 0; } void send_a_string(char *string_of_characters) { while(*string_of_characters > 0) { send_a_character(*string_of_characters++); } } |
Текст программы без комментариев
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
#include <avr/io.h> #define F_CPU 1000000 #include <util/delay.h> #include <stdlib.h> #define enable 5 #define registerselection 6 void send_a_command(unsigned char command); void send_a_character(unsigned char character); void send_a_string(char *string_of_characters); int main(void) { DDRB = 0xFF; DDRA = 0; DDRD = 0xFF; _delay_ms(50); ADMUX |=(1<<REFS0)|(1<<REFS1); ADCSRA |=(1<<ADEN)|(1<<ADATE)|(1<<ADPS0)|(1<<ADPS1)|(1<<ADPS2); int16_t COUNTA = 0; char SHOWA [3]; send_a_command(0x01); //Clear Screen 0x01 = 00000001 _delay_ms(50); send_a_command(0x38); _delay_ms(50); send_a_command(0b00001111); _delay_ms(50); ADCSRA |=(1<<ADSC); while(1) { COUNTA = ADC/4; send_a_string ("CIRCUIT DIGEST"); send_a_command(0x80 + 0x40 + 0); send_a_string ("Temp(C)="); send_a_command(0x80 + 0x40 + 8); itoa(COUNTA,SHOWA,10); send_a_string(SHOWA); send_a_string (" "); send_a_command(0x80 + 0); } } void send_a_command(unsigned char command) { PORTB = command; PORTD &= ~ (1<<registerselection); PORTD |= 1<<enable; _delay_ms(20); PORTD &= ~1<<enable; PORTB = 0; } void send_a_character(unsigned char character) { PORTB = character; PORTD |= 1<<registerselection; PORTD |= 1<<enable; _delay_ms(20); PORTD &= ~1<<enable; PORTB = 0; } void send_a_string(char *string_of_characters) { while(*string_of_characters > 0) { send_a_character(*string_of_characters++); } } |