Работа с датчиком температуры ds18b20 в bascom-avr

Работа с датчиком температуры DS18b20. Делаем термометр

Источник: http://AVRproject.ru/publ/kak_podkljuchit/termometr_na_ds18b20_mega32/2-1-0-15

Урок 12. Измерение температуры при помощи AVR. Простой термометр на AVR

Продолжаем осваивать периферию, на очереди измерение температуры. Рассмотрим вариант измерения, при помощи датчика температуры DS18b20.

Характеристики датчика: диапазон измерения от -55 до +125°С. Точность измерения ±0,5°С гарантируется в диапазоне от -10 до +85°С. Возможность измерения с разрешением 9, 10, 11 и 12 бит, т.е.

с шагом 0,5; 0,25; 0,125; 0,0625°С. Для обмена информацией с AVR микроконтроллером используется 1-Wire протокол.

Каждый датчик имеет свой уникальный адрес, поэтому имеется возможность посадить на шину сразу несколько датчиков.

Для сборки схемы понадобится жк дисплей, датчик и резистор на 4,7кОм. Теперь перейдем непосредственно к прошивке.

  Для измерения температуры в бытовых целях хорошо подходят цифровые датчики DS18B20. Эти датчики способны измерять температуру в пределах от -55 до 125 °C, с заявленной точностью от производителя в 0,5 °C.

Для большинства гражданских нужд такой диапазон и такая точность измерения вполне приемлема (какую-нибудь систему термоконтроля собрать или еще чего). Поэтому решил попрактиковаться с этим датчиком, обязательно пригодится.

 Работает датчик по однопроводному интерфейсу 1-Wire. Здесь я описывать этот протокол не буду, так как информации и так достаточно. Отличное описание работы интерфейса 1-wire и методика работы с датчиком ds18b20 можно почитать на http://samou4ka.net/

 Для нас самое важное, что нужно знать на первом этапе лишь то, что сигнальный провод необходимо подтягивать к шине питания подтягивающим резистором номиналом примерно 4,7к.

Если длина сигнального провода увеличивается, необходимо снижать сопротивление подтягивающего резистора.

Читал о том, что датчик соединяли с МК проводом длинной 90 метров, для исправной работы датчика сопротивление резистора составило 3к. 

  У себя для освоения работы с датчиком использовал микроконтроллер Atmega32 и выводил показания на двухстрочный алфавитно-цифровой ЖКИ. 

  Что касается программной части, в Bascom-AVR имеется библиотека работы с протоколом 1-Wire. Ее и задействуем. В программе полученные результаты с датчика немного корректируются, иначе при отрицательных температурах результат будет ложным на 1 градус, и нулевая температура выводится без знака.

$regfile = “m32def.dat”
$crystal = 1000000          ' 1 MHz

 '*** Конфигурируем ЖКИ

Config Lcd = 16 * 2
Config Lcdpin = Pin, Rs=Portc.5, E=Portc.4, Db4=Portc.0, Db5=Portc.1, Db6=Portc.2, Db7 = Portc.3

 ' ***Конфигурируем однопроводную шину

Config 1wire = Portb.7

 ' *** Заводим переменные

Dim Byte0 As Byte

Dim Byte1 As Byte
Dim Sign As String * 1
Dim T As Byte
Dim T1 As Byte
Dim T2 As Byte

Cls

Cursor Off  

 '***ОСНОВНАЯ ПРОГРАММА

Do                         ' начало цикла

1wreset

If Err = 1 Then            'если при опросе небыло ответа ставим флаг ошибки

Cls

Rem датчик

Rem не подключен
Lcd ” гaїАёє ”             ' выводим надпись об отсутствии датчика
Locate 2 , 1
Lcd ” Ѕe ѕoгє»ЖАeЅ”

Wait 1

Else                       ' иначе, если ошибки не было, продолжаем опрос датчика

1wwrite &HCC               ' Выдаем команду чтения ПЗУ

1wwrite &H44               ' Запуск измерения

Waitms 750                 ' Ждем окончания преобразования

1wreset

1wwrite &HCC
1wwrite &HBE               ' Команда чтения ОЗУ датчика

Byte0 = 1wread()           ' Читаем нулевой байт

Byte1 = 1wread()           ' Читаем первый байт

If Byte1 > 248 Then        ' Проверка на отрицательность температуры

Byte0 = &HFF – Byte0
Byte1 = &HFF – Byte1
Sign = “-“

Else

Sign = “+”
End If

T1 = Byte0 / 16           ' Сдвигаем нулевой байт вправо на 4 бита (2*2*2*2=16)

T2 = Byte1 * 16           ' Сдвигаем первый байт влево на 4 бита (2*2*2*2=16)

T1 = T1 + T2              ' Формируем результат для вывода на дисплей

 ' *** Коррекция полученных значений

If Sign = “-” Then        ' для корректного вывода отрицательных температур

 T1 = T1 + 1
End If

If Sign = “+” And T1 = 0 Then     ' убираем знак “+” с нулевой температуры

 Sign = ” “
End If

Cls

' *** Выводим результат на дисплей

Rem Температура

Lcd ” Teјѕepaїypa”                ' тут пишем что измеряем

Locate 2 , 6

Lcd Sign ; T1 ;                   ' выводим знак и температуру на дисплей
Locate 2 , 9

Lcd Chr(223) ; “C”                ' Сhr(223) – символ “градус”

Cursor Off

End If

Wait 1                            ' ждем 1 секундy

Loop                              ' повторяем цикл

End

  Датчик не подключен, выводится соответствующее сообщение:

  А это с подключенным датчиком, при комнатной температуре:

Ниже можете скачать исходник, файл в протеусе и откомпилированный файл:

Скачать

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
#include
#include
// 1 Wire Bus functions
#asm .equ __w1_port=0x18 ;PORTB .equ __w1_bit=2
#endasm
#include
#include
 
// Alphanumeric LCD Module functions
#asm .equ __lcd_port=0x12 ;PORTD
#endasm
#include
#include
 
char lcd_buf[17];
void main(void)
{
 
float temper; lcd_init(16); w1_init();
ds18b20_init(0,-20,50,DS18B20_12BIT_RES);  
while(1) { temper=ds18b20_temperature(0); sprintf(lcd_buf,”t=%.1fxdfC”,temper); lcd_clear(); lcd_puts(lcd_buf); delay_ms(1500); }; }

#include #include // 1 Wire Bus functions #asm .equ __w1_port=0x18 ;PORTB .equ __w1_bit=2 #endasm #include #include // Alphanumeric LCD Module functions #asm .

equ __lcd_port=0x12 ;PORTD #endasm #include #include char lcd_buf[17]; void main(void) { float temper; lcd_init(16); w1_init(); ds18b20_init(0,-20,50,DS18B20_12BIT_RES); while(1) { temper=ds18b20_temperature(0); sprintf(lcd_buf,”t=%.

1fxdfC”,temper); lcd_clear(); lcd_puts(lcd_buf); delay_ms(1500); }; }

Теперь обо всем по порядку:

1
2
3
4
#asm .equ __w1_port=0x18 ;PORTB .equ __w1_bit=2
#endasm

#asm .equ __w1_port=0x18 ;PORTB .equ __w1_bit=2 #endasm

Данный код означает, что датчик подключен к порту В, PB2 ножке

1
2
#include
#include

#include #include<\p>

Используется протокол 1wire, тип датчика ds18b20

1
2
3
float temper; w1_init();
ds18b20_init(0,-20,50,DS18B20_12BIT_RES);

float temper; w1_init(); ds18b20_init(0,-20,50,DS18B20_12BIT_RES);

Переменная temper (с плавающей точкой) используется для хранения температуры, w1_init(); — ищем датчик, ds18b20_init(0,-20,50,DS18B20_12BIT_RES); — настройка датчика: 0-номер датчика, -20 -нижний предел измерения, 50 — верхний предел измерения,

DS18B20_12BIT_RES используется 12 битный режим(с шагом 0,0625°С). В принципе настройку можно не производить, по умолчанию выставлен 12 битный режим. Показано лишь для того, чтобы вы могли самостоятельно изменить режим измерения, если это понадобится.

1
2
3
4
5
6
temper=ds18b20_temperature(0);   sprintf(lcd_buf,”t=%.1fxdfC”,temper); lcd_clear(); lcd_puts(lcd_buf); delay_ms(1500);

temper=ds18b20_temperature(0); sprintf(lcd_buf,”t=%.1fxdfC”,temper); lcd_clear(); lcd_puts(lcd_buf); delay_ms(1500);

temper=ds18b20_temperature(0); — читаем значение температуры с датчика sprintf(lcd_buf,»t=%.1fxdfC»,temper); преобразовываем к понятному для lcd виду %.1f — вывод числа с плавающей точкой 1 знак после запятой, не забываем в свойствах проекта указать (s)printf features float.

xdf — вывод на экран значка градуса.

В результате должно получиться нечто похожее

Отрицательной температуры поблизости не было :D, поэтому попробовал остудить бутылочкой соуса из холодильника, результат что то не сильно впечатлил.

Зато от нагрева рукой, температура довольно быстро повысилась.

Проект доступен тут
Проект для DS18s20
Проект для двух датчиков
Проект для DS18b20 на семисегментниках
Проект Алексея(Alyes)для Atmega16 и шести сегментов + бонус видео устройства

Источник: http://avr-start.ru/?p=796

Микроконтроллер AVR и датчик DS18B20

Доброго дня уважаемые друзья!
Приветствую Вас на сайте «Мир микроконтроллеров»

Последовательность операций для работы с датчиком DS18B20

При работе с датчиком DS18B20 очень важно выполнять последовательность выполнения операций:
1. Инициализация
2. Команда ROM
3. Функциональная команда
Если не следовать этой последовательности (что-то пропустить, или поменять местами), то датчик DS18B20 откажется общаться с микроконтроллером

Инициализация датчика DS18B20

В прошлой статье мы рассматривали этот вопрос (инициализация датчика DS18B20), но повторимся, что бы не прыгать по страницам сайта.

Процесс инициализации DS18B20 состоит из трех пунктов:
Исходное состояние — на выводе порта, к которому подключен вывод DQ датчика (шина DQ) — логическая 1 (за счет подтягивающего резистора)

1.

Микроконтроллер передает в линию импульс сброса (длительность 480 микросекунд): — переводит шину DQ в состояние логического нуля (вывод микроконтроллера переводится на вывод информации и в него записывается логический ноль) на время не менее 480 микросекунд (я обычно округляю длительность импульса до 500 микросекунд) — через 480 микросекунд (или 500) микроконтроллер переводит шину DQ на ввод информации, в результате чего за счет подтягивающего резистора на шине восстанавливается логическая единица

— микроконтроллер ждет импульс присутствия от датчика DS18B20

2. Датчик DS18B20 передает в линию импульс присутствия (общая длительность 480 микросекунд): — при переводе микроконтроллером шины в логический ноль на время не менее 480 микросекунд, датчик понимает, что к нему обращаются

— через 15-60 микросекунд, после того как МК перевел шину в состояние логической единицы, датчик передает импульс присутствия — переводит шину DQ в состояние логического нуля на время 60-240 микросекунд

3. Прием импульса присутствия: — через 15-60 микросекунд, после того как микроконтроллер перевел шину в состояние логической единицы, проверяется наличие логического нуля на шине DQ (брать лучше по максимуму — 60 микросекунд) — если на линии — логический ноль — значит датчик присутствует на шине и можно продолжать дальнейшую работу

— так-как общая длительность импульса присутствия должна быть не менее 480 микросекунд, микроконтроллер формирует дополнительную паузу длительностью 420 микросекунд

https://www.youtube.com/watch?v=ks4_XXK1B8w

В ходе процесса инициализации могут возникнуть три ошибки, которые мы разберем непосредственно в программе:

Процесс инициализации датчика DS18B20 оформляется в виде подпрограммы, в данном случае имя подпрограммы «DS_Reset_Pulse» В этой подпрограмме: — Term_Error1 — переменная, в которой будет хранится код ошибки

— DQ_Pin, DQ_Port, DQ_DDR — присвоенные мной имена выводу порта к которому подключен датчик.

Делается это не только для наглядности (можно конечно и ничего не присваивать) но в больше степени для быстрой смены вывода порта в программе, если мы вдруг захотим подключить датчик к другому выводу (не надо будет по всей программе искать где этот РВ6, чтобы заменить его, к примеру, на РС5 — проще один раз поменять в таблице)

Ну а теперь рассмотрим подробно саму подпрограмму «Инициализация датчика DS18B20» :

1.

Обнуляем переменную в которую будет записан код ошибки («Term_Error1») 2. Проверяем, есть ли на шине DQ логическая единица: — если на шине присутствует высокий уровень (логическая единица), то переходим к формированию импульса сброса — если на шине нет высокого уровня, то записываем в переменную «код ошибки» — 1 (код ошибки отсутствия высокого уровня на шине DQ) и выходим из подпрограммы. Затем, при выводе данных на индикатор, высвечиваем, к примеру, — «Er1».

Отсутствие высокого уровня в 99 случаях означает, что не подключен подтягивающий резистор формирующий высокий уровень на шине. 3. При высоком уровне на шине микроконтроллер формирует импульс сброса продолжительностью не мене 480 мкс, путем перевода вывода порта в режим «выхода» и записи в него логического нуля. 4. Микроконтроллер отпускает шину путем перевода вывода порта в режим «входа» 5.

Через 60 мкс МК поверяет наличие импульса присутствия от датчика (на шине должен присутствовать низкий уровень (логический ноль) 6. Проверяем, есть ли на шине DQ логический ноль: — при низком уровне на шине, означающем что датчик выдал импульс присутствия, МК формирует паузу в 420 мкс — высокий уровень на шине означает, что датчик не выдал импульс присутствия.

В этом случаем записываем в переменную «код ошибки» — 2 (код ошибки отсутствия импульса присутствия от датчика), а при выводе данных на индикатор — выводим «Er2» 7.

После паузы в 420 мкс проверяем наличие высокого уровня на шине после окончания импульса присутствия: — если на шине логическая единица, значит датчик отработал и выходим из подпрограммы инициализации датчика — если на шине логический ноль, значит датчик не вернул шину в высокий уровень. Записываем в переменную «код ошибки» — 3 (код ошибки «не восстановлен высокий уровень после импульса присутствия») На этом инициализация датчика заканчивается. Если ошибок нет, то МК переходит к выполнению следующих шагов — посылает на датчик команду ROM, а затем функциональную команду.

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

Команда ROM DS18B20

Мы рассмотрим работу микроконтроллера с одним датчиком температуры, это несколько проще в том смысле, что статья получится короче, ну а разобраться как работать с двумя и более датчиками будет, я думаю, не очень сложно.

Итак, мы выполнили первый шаг — инициализация датчика DS18B20, убедились что с датчиком все в порядке, ну а датчик приготовился к приему очередных команд и выдачи нужных нам результатов. Приступаем ко второму шагу — подаем нужную нам команду ROM.

Так как датчик у нас один, нам потребуется для работы одна команда ROM из пяти — SKIP ROM — пропуск ROM, так как мы не определяли 64-разрядный код датчика (для одного датчика этого делать не надо), соответственно отпадают и другие команды.

Практически, при одном датчике на шине, используется только одна команда ROM — SKIP ROM, и очень редко — ALARM SEARCH — Поиск Тревоги.

Перед тем как передать команду ROM на датчик, давайте разберемся как вообще передавать какие-либо данные по шине 1-Wire. Сразу скажу, самое главное в этом деле строго соблюдать временные интервалы.

Давайте посмотрим на временной график записи данных в датчик и на его основе составим подпрограмму записи данных в DS18B20:

Расшифровываем график: 1. Слот времени передачи одного бита должен иметь продолжительность не менее 60 мкс и не более 120 мкс 2. Датчик DS18B20 гарантированно определяет логический уровень на шине через 30 мкс (минимум — 15 мкс, максимум — 60 мкс) 3.

Так как датчику необходимо максимум 60 мкс для приема и записи бита информации, то делать продолжительность слота времени более 60 мкс смысла нет 3.

После окончания слота времени для передачи логического нуля необходимо не менее 1 мкс для перевода шины в высокое состояние (я беру 5 мкс) 4.

При передаче логической единицы в начале слота времени передачи МК должен перевести шину в низкое состояние на время от 1 до 15 мкс (я беру 5 мкс) Я в программе общения МК с датчиком процессы записи в датчик логических уровней оформил в виде 2-х подпрограмм: — запись логического нуля — запись логической единицы

Теперь посмотрите на мои подпрограммы и, исходя из выше сказанного, я думаю алгоритм передачи одного бита информации датчику будет понятен:

Теперь давайте посмотрим как передать байт информации датчику (все команды для датчика однобайтовые).
Запись байта информации в датчик также оформлена в виде подпрограммы:

Temp — рабочий регистр, в который предварительно записываем код передаваемой команды >Temp>> — операция логического сдвига вправо с переносом рабочего регистра Temp Передача байта информации от МК датчику производится побитно, начиная с младшего, т.е. сначала нулевой бит, затем первый и т.д.

Такую операцию очень удобно проводить с помощью команды «Логический сдвиг вправо с переносом», команда работает с регистрами общего назначения. При логическом сдвиге вправо с переносом, младший байт заносится в бит «С» регистра SREG, а содержимое РОН сдвигается вправо на 1 бит. Алгоритм работы подпрограммы: 1.

Задаем цикл, в ходе выполнения которого над регистром Temp будет восемь раз произведена операция логического сдвига вправо с переносом. В ходе выполнения цикла каждый бит регистра Temp, начиная с младшего «пройдет» через бит С регистра SREG 2.

После каждого логического сдвига с переносом проверяем что записалось в бит С регистра SREG: — если записан логический ноль — переходим к подпрограмме передачи нуля

— если записана логическая единица — переходим к подпрограмме передачи единицы.

Функциональные команды DS18B20

Как и команды ROM, функциональные команды однобайтовые и процесс их передачи датчику аналогичен процессу передачи команды ROM. В основном используются две функциональные команды: — «Конвертировать температуру», по этой команде датчик «измеряет температуру»

— «Чтение памяти», по этой команде датчик передает МК результаты конвертирования температуры

Конвертировать температуру (измерить температуру) DS18B20

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

Датчик в процессе конвертирования температуры удерживает шину DQ в низком состоянии, а по окончании конвертации переводит шину в высокое состояние.

МК должен отслеживать этот процесс — на шине логический ноль — значит идет конвертация, на шине логическая единица — значит датчик закончил конвертацию, и можно переходить к считыванию результатов измерения.

Посмотрим на примере алгоритм подачи команды на конвертирование температуры:

Как видно из приведенной программы, для измерения температуры датчиком, МК должен выполнить следующий алгоритм: 1. Инициализация датчика. Если возникла ошибка — выход из подпрограммы 2.

Подать команду «Пропуск ROM», где: — DS_SKIPROM — константа, в которой записан код команды $CC — для наглядности — DS_Command — переменная, в которую записываются команды от МК — DS_Send_Command — подпрограмма, в ходе выполнения которой код команды сначала записывается в рабочий регистр Temp, а затем происходит переход к подпрограмме передачи байта датчику.

Можно, конечно, присвоить имя DS_Command не переменной, а сразу рабочему регистру, или записывать код команды в рабочий регистр Temp, чтобы не делать лишних движений (это называется оптимизация кода). 3. Подать команду «Конвертировать температуру»

4. Перевести шину DQ на вход и постоянно проверять ее состояние — как только на шине будет логическая единица — выходим из подпрограммы.

Чтение памяти DS18B20

По команде «Чтение ROM» датчик передает на МК содержимое всей своей оперативной памяти — 9 байт информации, при этом результаты конвертирования находятся в первых двух байтах. Мы можем в любой момент времени прервать операцию чтения. К примеру, если нам нужны только данные измерения температуры — считываем первые два байта и прерываем процесс.

Для чтения результатов конвертирования температуры необходимо предварительно назначить две переменные, к примеру: — DS_Temp_Lo — для первого байта — DS_Temp_Hi — для второго байта

Посмотрим на подпрограмму чтения результатов конвертирования температуры датчиком (подпрограмма называется Read_Temp_1):

Алгоритм: 1. Инициализация датчика 2. Команда ROM — «Пропуск ROM» 3. Функциональная команда — «Чтение памяти» 4. Переход к подпрограмме чтения байтов МК (DS_Read_Byte)

5. Запись результатов в переменные

Рассмотрим как происходит чтение информации от датчика.
Временной график слотов времени передачи битов данных от датчика:

Расшифруем график: 1. Слоты времени считывания как логического 0, так и логической 1 должны быть продолжительностью не менее 60 мкс, и разделяться между собой промежутками не менее 1 мкс 2.

Для считывания бита информации МК переводит шину в низкое состояние на время не менее 1 мкс (у меня — 3 мкс), а затем возвращает шину в высокое состояние (Эта процедура является командой для датчика на передачу одного бита информации) 3.

Через 15 мкс датчик передает или логическую единицу, или логический ноль: — если логический 0 — датчик переводит шину в низкое состояние на время 45 мкс — если логическая 1 — датчик ничего ни делает, шина остается в состоянии логической единицы 4.

Через 15 мкс, после того, как МК перевел шину в низкое состояние, а затем вернул в высокое, в течение 45 мкс МК может считать данные с шины (у меня — через 23 мкс)

Давайте посмотрим подпрограмму чтения байта:

Подпрограмма чтения байта от датчика очень похожа на подпрограммы передачи данных от МК. Также используется цикл и команда логического сдвига вправо с переносом регистра Temp, датчик данные передает начиная с нулевого бита. Давайте посмотрим 1 цикл выполнения подпрограммы: 1.

МК переводит шину МК в низкое состояние на 3 мкс, а затем обратно в высокое 2. Через 20 мкс МК анализирует логический уровень на шине: — если на шине низкий уровень — в бит С регистра SREG записывается 0 — если на шине выокий уровень — в бит С регистра SREG записывается 1 3.

Производится логический сдвиг вправо с переносом регистра Temp, при этом содержимое бита С регистра SREG записывается в старший бит Temp.

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

Считанные байты мы записываем в назначенные переменные. Хочу обратить ваше внимание на то, что в первом байте результатов конвертирования температуры, в первых четырех битах записаны «Десятые градуса» в виде какого-то числа.

К примеру, при 12-битной разрешающей способности датчика, это число может быть от 0 до 14, поэтому десятые градуса рассчитываются самостоятельно.

Старшие 4 бита 1-го байта и младшие четыре байта 2 байта содержат информацию о целых градусах именно в градусах.

Это не очень удобно для работы, и поэтому сначала надо выделить «десятые» в отдельный байт путем записи первых 4-х битов 1-го байта в отдельную переменную. Старшие 4 бита 1-го байта записываем в младшие 4 бита другой переменной, и в эту же переменную, в старшие 4 бита, записываем младшие 4 бита 2-го байта. К примеру так (заодно и определяем знак температуры):

И последнее, если температура отрицательная, то данные в датчике хранятся в виде дополнительного кода.

Чтобы привести их в «нормальное состояние», необходимо, после того как выделим «десятые» и «целые» градуса в разные байты, проинвертировать их содержимое — 0 поменять на 1, а 1 — на 0, а затем, к полученному результату прибавить единицу. Для этого удобно применить двойной регистр (X, Y или Z).

Предыдущие статьи:

1. Устройство цифрового термометра DS18B20
2. Система команд датчика DS18B20
3. Подключение DS18B20 к микроконтроллеру

Источник: https://microkontroller.ru/praktikum-mikrokontrollershhika/programmirovanie-rabotyi-datchika-ds18b20-s-mikrokontrollerom/

Работа с датчиком температуры DS18B20

avrki@avrki.ru

Дата: 15 Мая 2014. Автор: Алексей

В библиотеке AXLIB есть набор функций для работы с данным датчиком. Весна в полном разгаре. Скоро лето, жара. А жара это сколько? Надо померить. А как? Вот и решил я поработать с датчиком температуры DS18B20. Благо он дешевый и его можно купить где угодно. И еще под этот датчик в CVAVR есть библиотека. Но….

После ковыряния с этим датчиком и библиотекой от CVAVR я окончательно решил слезть с этой пакости. Я пол дня угробил на состыковку библиотечных функция с динамической индикацией. Все вроде работает нормально, но когда доходит до опроса датчика, то индикация спотыкается на секунду.

Жуть!!! Короче пообщавшись с умными людьми на форуме, я залез внутрь библиотеки и нарыл тело функции чтения температуры и о чудо! Там за каким-то лешим после команды на старт преобразования стоит пауза на 550 мс. Зачем? Для тех кто все же останется преданным CVAVR, эту паузу можно стереть. Заработало немного быстрее.

Еще остался вопрос, а где лежат функции для работы с шиной 1-wire? Только хедер и все. Просто возник еще вопрос. Если при динамической индикации использовать таймер и в прерывании выводить цифры, то фигня выходит. Терзают меня смутные сомнения, что при обработке функций шины 1-wire запрещаются прерывания. Короче мне эти непонятки надоели и я переехал на AVRStudio 6.

1 и ни капли не сожалею. Ну и еще получил пинок для изучения принципа работы шины 1-wire. Перед тем как писать код, давайте окунемся немного в теорию. Сразу оговорюсь, я рассчитываю на то что статьи такого типа читают люди которые чуть ли не впервые взяли в руки паяльник. И если начать трактат о временных слотах и тому подобное, то многие прокиснут и скажут, а ну его на фиг.

Поэтому я буду объяснять на пальцах как сам это понимал.
Для начала определимся с планом действий. Задача: Один датчик, берем из него температуру и по необходимости выводим куда-нибудь. Алгоритм из мануала на датчик.

  • 1. Подать команду сброса.
  • 2. Получить подтверждение о наличии хотя бы одного датчика.
  • 3. Читать РОМ код (эта фигня с его идентификационным номером. Нужно для работы не с одним датчиком).
  • 4. Передать настройки
  • 5. Подать команду на начало преобразования температуры
  • 6. Получить данные
  • 7. Преобразовать в человеческий вид

Теперь все по порядку. Перед тем как начать рассказ о командах, давайте пробежимся по физике шины.

У датчика есть три ноги, две для питания(возможна работа от паразитного питания, но о ней мы пока говорить не будем. Считаем что у нас есть полноценное питание и оно подведено к датчику). Третья нога как раз для передачи данных. Но тут есть свой нюанс. Просто дергать ногу верх и вниз нельзя.

Между этой ножкой и плюсом питания стоит резистор на 4,7К, который подтягивает ногу к плюсу питания. То есть если мы шину не трогаем(нет передачи) то на линии висит плюс питания. Также надо запомнить, что все начала передач инициируем мастер. Датчик всегда сидит и нюхает шину, а как только там что-то появилось, отвечает. Подаем команду сброса.

Для того чтобы подать команду сброса, нужно прижать линию данных к нулю и удержать в таком состоянии не менее 480 мкс. Потом просто отпустить линию. Тот резистор, о котором я говорил ранее, подтянет шину к плюсу питания. Далее после того как мы отпустили шину ждем 15 мкс и смотрим.

Если шина упала к нулю и так продержалась не менее 60 мкс, то значит на шине висит датчик и он нам ответил ” Я тут”. Если на шине после выжидания 15 мкс ноль не появился, значит на шине кроме мастера никого больше нет. Вот и вся команда.

Если на шине никого нет, то забиваем на все и занимаемся чем-нибудь другим, а вот если кто-то есть, а скорее всего наш датчик, то мы имеем полное право ему дать команду. Следующая команда по очереди должна быть чтение РОМ кода. Что такое РОМ код. Это группа из 8 байтов которая несет в себе некую информацию, а именно: Младший байт содержит код семейства.

В мануале написано что равен 0х28, но я почему-то получал 0х10. Следующие 6 байтов это идентификационный номер датчика. Как я понял он уникален или фирма выпустила больше 281474976710655 датчиков. Я даже не знаю как это число произнести)). И последний байт это CRC контрольная сумма предыдущих байт. Получить ее можно подав команду 0х33. Но тут тоже не все так просто.

Команда 0х33 это запрос РОМ кода если датчик один, а чтоб выудить их из кучи датчиков, то нужно сказать 0хF0. Мудрено? Да… Но если вспомнить что мы хотим общаться с одним датчиком, то нам этот РОМ код ваще читать не надо. Но мануал требует обязательного чтения РОМ кода. Что делать? А вот что. Есть такая команда под названием “Гей хлопцы се кричим!”. То есть пишем 0хСС.

Это означает пропустить РОМ код и всем устройствам на шине принять команду. В научном мире это называется широковещательная передача. То есть не важно сколько там устройств на шине, следующая после 0хСС команда одресованна всем.
Замечательно, подготовили устройство к получению команды и оно напряглось. Что дальше.

А дальше нам нужно сказать ему “А нука друг измерий ка мне температурку. И шлем ему код 0х44. Тот получив приказ, тут же подрывается на работу. Вопрос. А как узнать когда датчик измерил температуру? А просто. Пока он меряет, то прижимает шину к нулю. Отсюда мы нюхаем шину и как только получили единицей по носу, о! Измерил.
Получение температуры. Датчик туп как пробка.

Получив приказ измерить температуру он это сделал и положил данные в память. А дальше выпучив глаза ждет очередной команды. Мы в ту же очередь, получив единицей по носу, говорим 0хВЕ. Датчик получив эту команду понимает что надо передать данные из памяти. А какие? Память датчика состоит из 9 байт. Первые два это температура.

Потом два байта тревог по превышению верхнего или нижнего предела температуры.(некий термостат). Следующий байт это конфигурационный регистр. Ну мол как цифровать температуру. Далее три байта в резерве. И наконец последний CRC. Помним что это). Так вот при подачи команды 0хВЕ датчик не будет думать о том что надо передать, а просто вывалит все девять байт в шину.

Но мы все получать не будем. Мы получим только первые два байта. Ведь мы работаем только с одним датчиком и нас интересует только температура. Получив первые два байта переходим к их дешифровке, а остальные байты пускай затеряются на шине.
Преобразование данных в человеческий вид. Получив два байта нам нужно сделать из них температуру.

Ну для начала надо понять что температура бывает как положительной, так и отрицательной. А с учетом того что у нас два байта, то и выделять под переменную нам надо типа int. Далее вроде все просто. Младший байт в младший инта, а старший в старший. Да не все так просто. Я долго вкуривал в это безумие и так и не понял)) Но всеми известный научный метод тыка пока не кто не отменял.

Короче берем старший байт и кладем его как есть в старший байт инта, а младший сначала умножаем на 5, а потом кладем в младший байт того же инта. Все, вот наша температура правда умноженная на 10. То есть если температура 23,5 гр. то в инте будет 235. А если температура отрицательная? Во подстава.

Глядя в таблицу температур датчика я понял что ее передают в, как по научному вроде, обратной нотации что ли. Короче, после получения данных от датчика, нужно проверить старший бит у старшего байта на 0. Если ноль, значит плюс, а если единица, значит зима) А дальше склеиваем как плюсовую, а потом инвертируем каждый бит. То есть те что были нулями должны стать единицами, а единицы нулями. После прибавляем единицу и получаем температуру так сказать по модулю. Выводим как плюсовую, только не забываем поставить знак минус.
Ну вроде и все. Вот фотки плюсовой и минусовой температуры. Правда минус не совсем низкий так как сейчас за окном май месяц, а от куска мяса из морозилки много не выдавишь но все же работу программы надо показать)
Программный код я здесь выкладывать не буду, так как это лишнее. Я выкладываю архив в котором лежат файлы основной программы и подключаемого хедера с функциями передачи команд. Каждая строка подробно закоментированна. Если все же возникнут вопросы, то я на связи. Удачи! В библиотеке AXLIB есть набор функций для работы с данным датчиком.

Архив с файлами.



Алексей, вопрос – я могу с помощью команды читать ROM, определить наличие датчика? Скажем, выдать 0x33, и прочитать в первом байте код семейства DS18B20. Если он не равен 0x28 – значит, датчика нет. Или так не не годится?

А чем не устраивает команда сброса? Или на шине еще что-то висит?

Но, ведь по команде сброса можно только определить – есть ли любые устройства 1- wire или нет. А хотелось бы именно удостовериться в отсутствии термодатчика DS18B20.

Есть чужой исходник на ассемблере на два термометра с батарейным питанием на разных шинах, захотелось кое-что переделать по своему вкусу. Лезть в сами процедуры работы с датчиком особо не хочется, вроде нормально и так работает.

Поэтому и возникла идея – дать команду 0x33, потом принять первый байт из 8, и его проанализировать на наличие кода семейства. Типа, если там есть 0x28 –

Хорошо. Если есть два датчика НО НА РАЗНЫХ шинах, то какая разница как определить наличие датчика. Подали команду сброса и все. Датчик ответил, значит все в порядке. Не ответил, увы. Команды на обнаружение по семейству нужно подавать только тогда, когда на шине висит больше одного датчика. А если один, то зачем такие сложности. Или я что-то не понимаю в вашем вопросе.

Спасибо за ответ. Просто хотел уточнить, возможен ли такой вариант определения наличия датчика.

Собственно, а почему нет. Если известен адрес датчика и он ответил на любую команду значит он априори жив и здоров.

Меня как раз интересовал способ, при котором неизвестен адрес датчика, сидящего на отдельной шине. Зачем полностью читать все 64 бита идентификации, вполне хватит и 8-ми бит, в которых есть код семейства. Например, 0x28 для DS18B20. Если принят любой другой код – значит, датчика нет, он не тот, или неисправен, что в конкретном случае почти одинаково.

Стоп, стоп, стоп. Это сработает при широковещательном запросе при условии только одного датчика на шине. С тем же успехом можно и команду сброса подать.

Если будет больше устройств на шине, то на широковещательный запрос ответят все. Иначе по адресу. Ну это же логично. Если вы отправите письма с запросом нескольким людям, они все вам и ответят.

А если просто одному человеку в комнате крикните, “Эй, ты!”. Он один и ответит.

Правильно, я и написал – “…датчика, сидящего на отдельной шине…”. У меня как раз ситуация, когда каждый датчик термометра – сидит на отдельной шине. Именно поэтому и возник такой вопрос. И потом, стоит задача определения наличия именно DS18B20, мало ли что там народ еще может подключить. А так – датчик ответил кодом 0x28, значит все в порядке.

Алексей, еще поясните такую вещь, если не трудно, а то что-то в даташите про это нигде не сказано.

Я подал команду записать в датчик 0x33 (чтение ROM), затем должен сколько-то подождать, или могу сразу читать с датчика? Где-то в сети народ жаловался на то, что типа, пока не поставил между командами записи-чтения задержку около 1 миллисекунды – датчик не отзывался вовсе, или вывдавал всякую белиберду.

А это зависит от дискретизации(точности) измерений. При запросе на вычисления температуры датчик может, зависать аж по моему до 750 мс. Давненько я с ним не работал не помню. Проще шину нюхать.

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

С этой проблемой часто сталкиваются те кто выводит температуру на динамический индикатор.

Когда нужна температура, задержка обычно подразумевается. Потому что делать поллинг (опрос бита готовности), не всегда удобно. А вот когда пишется команда, не связанная с преобразованием температуры? Например, та же 0x33 – чтение ROM.

Ну я не знаю. Вообще задержка не нужна. Ради собственного удовлетворения то пару миллисекунд вляпать можно. Если это не дипломная работа, то я бы воспользовался импирикой и не парил себе мозг. 🙂

Есть желание немножко усложнить конструкцию? Я очень хочу поработать с вами вместе. Передо мной была поставлена интересная задача. Денежкой поделюсь. Жду ответа тут

А по конкретнее какая задача. Точнее ТЗ.

Нужно сделать два термометра МК1 и МК2. На мк1 навшано BT модуль HC-05, ds18b20,радио NRF24L01. На мк2 ds18b20,радио NRF24L01.

—————-мк1—————– мк 1 измеряет temp_mk1 пишем в РОН инициируем подключение по радио NRF, если есть подключение, кидаем ему общую уставку Y и просим передать temp_mk2 c MK2 (пишем в POH) -инициируем подключение по БТ,если есть отправляем temp_mk1 и temp_mk2 и просим получить уставку Y(получаем и копируем в РОН) – считаем ошибку температуры мк1, если отрицательная включаем реле. ————-Мк2—————- Измеряем температуру и пишем в РОН слушаем эфир NRF, если есть отправляем _температуру и просим уставку. Сравниваем, результат в ошибку. Если отрицательная, включаем реле. Это регулятор для фуры отца. Будет греть прицеп. Говорит, мол не обойтись. Платит мне 10К а я все нее могу с модулями и ассемблером разобраться. Нашел ваш терморегулятор. Разбираю ваш пример пока. Изложение мысли понравилось.

Такой регулятор…Как связаться с вами?

Отлаживаю на плате STK500 в связке с STK502. Примерную схему я набросал в ИГЛКАД Для связи могу предложить скайп. Мой логин : 819 готов отдать 7К

Честно, я ничего не понял. avrki@avrki.ru моя почта. Можно более подробно и сформулировано объяснить что нужно сделать. И при чем здесь РОН? Обязательно на ассемблере? Какой МК рулить всем этим будет? Нужна конкретика, ТЗ. Напишите поверхностный псевдо-код, что должно происходить в течении работы устройства. На какой шине висят радио модуль и ВТ модуль. Зачем вообще ВТ? Связь с телефоном?

Здравствуйте, а скажите насколько точно он меряет температуру. У меня он по ощущениям на 3-4 градуса завышает. Минусовую вроде точно выводит. Также хочу заметить, что нет смысла выводить десятые доли градуса при такой точности. Тогда температуру можно получать так temper = (temp_0>>4)+(temp_1

Источник: http://www.avrki.ru/articles/content/ds18b20/

Скачать и слушать бесплатно музыку mp3

Улети by T-Fest

Между нами ток by Татьяна Котова

Tell Me Who (feat. Eneli) [Slider & Magnit Remix] by Vanotek

Got U (Ready or Not) by Kokab

Сегодня я нашел тебя by Jah Khalib

Despacito (feat. Daddy Yankee) by Luis Fonsi

I Got Love (feat. Рем Дигга) by MiyaGi & Эндшпиль

Thunder by Imagine Dragons

Attention by Charlie Puth

Stole the Show (feat. Parson James) by Kygo

случайная by Loboda

Моя любовь by Макс Барских

Сансара (feat. Диана Арбенина, Александр Ф. Скляр, Сергей Бобунец, Sunsay, Скриптонит & Ант) by Баста

Львица by Lx24

Fetish (feat. Gucci Mane) by Selena Gomez

Believer by Imagine Dragons

Подсыпал by Елена Темникова

Одна на миллион by Наталья ВОВК

Между нами любовь by SEREBRO

Неделимы by Artik & Asti

Лейла (feat. Маквин) by Jah Khalib

Numb by LINKIN PARK

Дни и ночи by Джиган

Тает лёд by Грибы

На двоих by МакSим

ОТПУСТИЛ by ВЕТЕР

Потрачу by Егор Крид

Вдох by Елена Темникова

По волнам by Burito

#Тамада (feat. al l bo & Wooshendoo) by Miyagi & Эндшпиль

Move On (feat. Jabbar) by Deeperise

Vitamin D by MONATIK

Драмы больше нет by Полина Гагарина

Твои глаза by Loboda

Если ты меня не любишь by Егор Крид & MOLLY

Way Down We Go by Kaleo

Малополовин by Ольга Бузова

Shape of You by Ed Sheeran

Если чё, я Баха by Jah Khalib

Лирика (feat. Masha) by Filatov & Karas

In the End by LINKIN PARK

Feel It Still by Portugal. The Man

Love by Lana Del Rey

Туманы by Макс Барских

Talking to Myself by LINKIN PARK

One More Light by LINKIN PARK

Пахну лишь тобой (feat. Artik & Asti) by Глюк'oZa

УВЛИУВТ (Упали в любовь и ударились в танцы) by MONATIK

2U (feat. Justin Bieber) by David Guetta

Tell Me Who (feat. Eneli) by Vanotek

Источник: https://mp3-muzyka.ru/download/WSj1dMP3b-s/%D0%A3%D1%80%D0%BE%D0%BA-15-%D0%A0%D0%B0%D0%B1%D0%BE%D1%82%D0%B0-%D1%81-%D0%B4%D0%B0%D1%82%D1%87%D0%B8%D0%BA%D0%BE%D0%BC-%D1%82%D0%B5%D0%BC%D0%BF%D0%B5%D1%80%D0%B0%D1%82%D1%83%D1%80%D1%8B-DS18B20-%D0%B2-BASCOM-AVR.html

Подключение датчика ds18b20 к микроконтроллеру ATmega8535

Источник: http://mega-avr.ucoz.ru/publ/ds18b20_connect_atmega8535/1-1-0-15

Ссылка на основную публикацию
Adblock
detector
",css:{backgroundColor:"#000",opacity:.6}},container:{block:void 0,tpl:"
"},wrap:void 0,body:void 0,errors:{tpl:"
",autoclose_delay:2e3,ajax_unsuccessful_load:"Error"},openEffect:{type:"fade",speed:400},closeEffect:{type:"fade",speed:400},beforeOpen:n.noop,afterOpen:n.noop,beforeClose:n.noop,afterClose:n.noop,afterLoading:n.noop,afterLoadingOnShow:n.noop,errorLoading:n.noop},o=0,p=n([]),h={isEventOut:function(a,b){var c=!0;return n(a).each(function(){n(b.target).get(0)==n(this).get(0)&&(c=!1),0==n(b.target).closest("HTML",n(this).get(0)).length&&(c=!1)}),c}},q={getParentEl:function(a){var b=n(a);return b.data("arcticmodal")?b:(b=n(a).closest(".arcticmodal-container").data("arcticmodalParentEl"),!!b&&b)},transition:function(a,b,c,d){switch(d=null==d?n.noop:d,c.type){case"fade":"show"==b?a.fadeIn(c.speed,d):a.fadeOut(c.speed,d);break;case"none":"show"==b?a.show():a.hide(),d();}},prepare_body:function(a,b){n(".arcticmodal-close",a.body).unbind("click.arcticmodal").bind("click.arcticmodal",function(){return b.arcticmodal("close"),!1})},init_el:function(d,a){var b=d.data("arcticmodal");if(!b){if(b=a,o++,b.modalID=o,b.overlay.block=n(b.overlay.tpl),b.overlay.block.css(b.overlay.css),b.container.block=n(b.container.tpl),b.body=n(".arcticmodal-container_i2",b.container.block),a.clone?b.body.html(d.clone(!0)):(d.before("
"),b.body.html(d)),q.prepare_body(b,d),b.closeOnOverlayClick&&b.overlay.block.add(b.container.block).click(function(a){h.isEventOut(n(">*",b.body),a)&&d.arcticmodal("close")}),b.container.block.data("arcticmodalParentEl",d),d.data("arcticmodal",b),p=n.merge(p,d),n.proxy(e.show,d)(),"html"==b.type)return d;if(null!=b.ajax.beforeSend){var c=b.ajax.beforeSend;delete b.ajax.beforeSend}if(null!=b.ajax.success){var f=b.ajax.success;delete b.ajax.success}if(null!=b.ajax.error){var g=b.ajax.error;delete b.ajax.error}var j=n.extend(!0,{url:b.url,beforeSend:function(){null==c?b.body.html("
"):c(b,d)},success:function(c){d.trigger("afterLoading"),b.afterLoading(b,d,c),null==f?b.body.html(c):f(b,d,c),q.prepare_body(b,d),d.trigger("afterLoadingOnShow"),b.afterLoadingOnShow(b,d,c)},error:function(){d.trigger("errorLoading"),b.errorLoading(b,d),null==g?(b.body.html(b.errors.tpl),n(".arcticmodal-error",b.body).html(b.errors.ajax_unsuccessful_load),n(".arcticmodal-close",b.body).click(function(){return d.arcticmodal("close"),!1}),b.errors.autoclose_delay&&setTimeout(function(){d.arcticmodal("close")},b.errors.autoclose_delay)):g(b,d)}},b.ajax);b.ajax_request=n.ajax(j),d.data("arcticmodal",b)}},init:function(b){if(b=n.extend(!0,{},a,b),!n.isFunction(this))return this.each(function(){q.init_el(n(this),n.extend(!0,{},b))});if(null==b)return void n.error("jquery.arcticmodal: Uncorrect parameters");if(""==b.type)return void n.error("jquery.arcticmodal: Don't set parameter \"type\"");switch(b.type){case"html":if(""==b.content)return void n.error("jquery.arcticmodal: Don't set parameter \"content\"");var e=b.content;return b.content="",q.init_el(n(e),b);case"ajax":return""==b.url?void n.error("jquery.arcticmodal: Don't set parameter \"url\""):q.init_el(n("
"),b);}}},e={show:function(){var a=q.getParentEl(this);if(!1===a)return void n.error("jquery.arcticmodal: Uncorrect call");var b=a.data("arcticmodal");if(b.overlay.block.hide(),b.container.block.hide(),n("BODY").append(b.overlay.block),n("BODY").append(b.container.block),b.beforeOpen(b,a),a.trigger("beforeOpen"),"hidden"!=b.wrap.css("overflow")){b.wrap.data("arcticmodalOverflow",b.wrap.css("overflow"));var c=b.wrap.outerWidth(!0);b.wrap.css("overflow","hidden");var d=b.wrap.outerWidth(!0);d!=c&&b.wrap.css("marginRight",d-c+"px")}return p.not(a).each(function(){var a=n(this).data("arcticmodal");a.overlay.block.hide()}),q.transition(b.overlay.block,"show",1*")),b.overlay.block.remove(),b.container.block.remove(),a.data("arcticmodal",null),n(".arcticmodal-container").length||(b.wrap.data("arcticmodalOverflow")&&b.wrap.css("overflow",b.wrap.data("arcticmodalOverflow")),b.wrap.css("marginRight",0))}),"ajax"==b.type&&b.ajax_request.abort(),p=p.not(a))})},setDefault:function(b){n.extend(!0,a,b)}};n(function(){a.wrap=n(document.all&&!document.querySelector?"html":"body")}),n(document).bind("keyup.arcticmodal",function(d){var a=p.last();if(a.length){var b=a.data("arcticmodal");b.closeOnEsc&&27===d.keyCode&&a.arcticmodal("close")}}),n.arcticmodal=n.fn.arcticmodal=function(a){return e[a]?e[a].apply(this,Array.prototype.slice.call(arguments,1)):"object"!=typeof a&&a?void n.error("jquery.arcticmodal: Method "+a+" does not exist"):q.init.apply(this,arguments)}}(jQuery)}var debugMode="undefined"!=typeof debugFlatPM&&debugFlatPM,duplicateMode="undefined"!=typeof duplicateFlatPM&&duplicateFlatPM,countMode="undefined"!=typeof countFlatPM&&countFlatPM;document["wri"+"te"]=function(a){let b=document.createElement("div");jQuery(document.currentScript).after(b),flatPM_setHTML(b,a),jQuery(b).contents().unwrap()};function flatPM_sticky(c,d,e=0){function f(){if(null==a){let b=getComputedStyle(g,""),c="";for(let a=0;a=b.top-h?b.top-h{const d=c.split("=");return d[0]===a?decodeURIComponent(d[1]):b},""),c=""==b?void 0:b;return c}function flatPM_testCookie(){let a="test_56445";try{return localStorage.setItem(a,a),localStorage.removeItem(a),!0}catch(a){return!1}}function flatPM_grep(a,b,c){return jQuery.grep(a,(a,d)=>c?d==b:0==(d+1)%b)}function flatPM_random(a,b){return Math.floor(Math.random()*(b-a+1))+a}
");let k=document.querySelector(".flat_pm_modal[data-id-modal=\""+a.ID+"\"]");if(-1===d.indexOf("go"+"oglesyndication")?flatPM_setHTML(k,d):jQuery(k).html(b+d),"px"==a.how.popup.px_s)e.bind(h,()=>{e.scrollTop()>a.how.popup.after&&(e.unbind(h),f.unbind(i),j())}),void 0!==a.how.popup.close_window&&"true"==a.how.popup.close_window&&f.bind(i,()=>{e.unbind(h),f.unbind(i),j()});else{let b=setTimeout(()=>{f.unbind(i),j()},1e3*a.how.popup.after);void 0!==a.how.popup.close_window&&"true"==a.how.popup.close_window&&f.bind(i,()=>{clearTimeout(b),f.unbind(i),j()})}f.on("click",".flat_pm_modal .flat_pm_crs",()=>{jQuery.arcticmodal("close")})}if(void 0!==a.how.outgoing){let b,c="0"==a.how.outgoing.indent?"":" style=\"bottom:"+a.how.outgoing.indent+"px\"",e="true"==a.how.outgoing.cross?"":"",f=jQuery(window),g="scroll.out"+a.ID,h=void 0===flatPM_getCookie("flat_out_"+a.ID+"_mb")||"false"!=flatPM_getCookie("flat_out_"+a.ID+"_mb"),i=document.createElement("div"),j=jQuery("body"),k=()=>{void 0!==a.how.outgoing.cookie&&"false"==a.how.outgoing.cookie&&h&&(jQuery(".flat_pm_out[data-id-out=\""+a.ID+"\"]").addClass("show"),j.on("click",".flat_pm_out[data-id-out=\""+a.ID+"\"] .flat_pm_crs",function(){flatPM_setCookie("flat_out_"+a.ID+"_mb",!1)})),(void 0===a.how.outgoing.cookie||"false"!=a.how.outgoing.cookie)&&jQuery(".flat_pm_out[data-id-out=\""+a.ID+"\"]").addClass("show")};switch(a.how.outgoing.whence){case"1":b="top";break;case"2":b="bottom";break;case"3":b="left";break;case"4":b="right";}jQuery("body > *").eq(0).before("
"+e+"
");let m=document.querySelector(".flat_pm_out[data-id-out=\""+a.ID+"\"]");-1===d.indexOf("go"+"oglesyndication")?flatPM_setHTML(m,d):jQuery(m).html(e+d),"px"==a.how.outgoing.px_s?f.bind(g,()=>{f.scrollTop()>a.how.outgoing.after&&(f.unbind(g),k())}):setTimeout(()=>{k()},1e3*a.how.outgoing.after),j.on("click",".flat_pm_out .flat_pm_crs",function(){jQuery(this).parent().removeClass("show").addClass("closed")})}countMode&&(flat_count["block_"+a.ID]={},flat_count["block_"+a.ID].count=1,flat_count["block_"+a.ID].click=0,flat_count["block_"+a.ID].id=a.ID)}catch(a){console.warn(a)}}function flatPM_start(){let a=flat_pm_arr.length;if(0==a)return flat_pm_arr=[],void jQuery(".flat_pm_start, .flat_pm_end").remove();flat_body=flat_body||jQuery("body"),!flat_counter&&countMode&&(flat_counter=!0,flat_body.on("click","[data-flat-id]",function(){let a=jQuery(this),b=a.attr("data-flat-id");flat_count["block_"+b].click++}),flat_body.on("mouseenter","[data-flat-id] iframe",function(){let a=jQuery(this),b=a.closest("[data-flat-id]").attr("data-flat-id");flat_iframe=b}).on("mouseleave","[data-flat-id] iframe",function(){flat_iframe=-1}),jQuery(window).on("beforeunload",()=>{jQuery.isEmptyObject(flat_count)||jQuery.ajax({async:!1,type:"POST",url:ajaxUrlFlatPM,dataType:"json",data:{action:"flat_pm_ajax",data_me:{method:"flat_pm_block_counter",arr:flat_count}}})}).on("blur",()=>{-1!=flat_iframe&&flat_count["block_"+flat_iframe].click++})),flat_userVars.init();for(let b=0;bflat_userVars.textlen||void 0!==a.chapter_sub&&a.chapter_subflat_userVars.titlelen||void 0!==a.title_sub&&a.title_subc&&cc&&c>d&&(b=flatPM_addDays(b,-1)),b>e||cd||c-1!=flat_userVars.referer.indexOf(a))||void 0!==a.referer.referer_disabled&&-1!=a.referer.referer_disabled.findIndex(a=>-1!=flat_userVars.referer.indexOf(a)))&&(c=!0),c||void 0===a.browser||(void 0===a.browser.browser_enabled||-1!=a.browser.browser_enabled.indexOf(flat_userVars.browser))&&(void 0===a.browser.browser_disabled||-1==a.browser.browser_disabled.indexOf(flat_userVars.browser)))){if(c&&void 0!==a.browser&&void 0!==a.browser.browser_enabled&&-1!=a.browser.browser_enabled.indexOf(flat_userVars.browser)&&(c=!1),!c&&(void 0!==a.geo||void 0!==a.role)&&(""==flat_userVars.ccode||""==flat_userVars.country||""==flat_userVars.city||""==flat_userVars.role)){flat_pm_then.push(a),flatPM_setWrap(a),flat_body.hasClass("flat_pm_block_geo_role")||(flat_body.addClass("flat_pm_block_geo_role"),flatPM_ajax("flat_pm_block_geo_role")),c=!0}c||(flatPM_setWrap(a),flatPM_next(a))}}}let b=jQuery(".flatPM_sticky");b.each(function(){let a=jQuery(this),b=a.data("height")||350,c=a.data("top");a.wrap("
");let d=a.parent()[0];flatPM_sticky(this,d,c)}),debugMode||countMode||jQuery("[data-flat-id]:not([data-id-out]):not([data-id-modal])").contents().unwrap(),flat_pm_arr=[],jQuery(".flat_pm_start, .flat_pm_end").remove()}

В комплекте с CodeVisionAVR имеется библиотека ds18b20.h ( соответственно в составе с ds18b20.lib ), которая позволяет работать с датчиком температуры ds18b20 – основная задача, заключается в предоставлении значения измеренной температуры. В этой статье поговорим о некоторых, наиболее применимых функциях библиотеки.

Небольшой обзорDS18B20 – это цифровой термометр с программируемым разрешением, от 9 до 12–bit, которое может сохраняться в EEPROM памяти прибора. DS18B20 обменивается данными по 1-Wire шине и при этом может быть как единственным устройством на линии так и работать в группе. Все процессы на шине управляются центральным микропроцессором.DS18B20 состоит из ПЗУ содержащее 64-битный последовательный код, который позволяет, общаться с множеством датчиков DS18B20 установленных на одной шине, контроллера MicroLAN, температурного датчика, двух регистров для хранения верхнего и нижнего порогов температуры и регистра конфигурации. Регистр конфигурации позволяет пользователю устанавливать разрешающую способность цифрового преобразователя температуры к 9, 10, 11, или 12 битам, это и влияет на время конвертирования температуры.Термометр не содержит внутреннего источника, а использует “паразитное” питание от однопроводной шины. Однако при измерении температуры и записи данных в ЭППЗУ ток потребления микросхемы превышает 1 мА, в то время как максимальный ток, который может обеспечить ведущий шины с помощью нагрузочного резистора 1,5…5 кОм, составляет 3,3…1 мА. Применение внешнего источника питания ускоряет преобразование температуры, поскольку от ведущего шины не требуется ожидания в течение максимально возможного времени преобразования. В этом случае все приборы DS18B20, расположенные на шине, могут выполнять преобразование температуры одновременно и во время обмена данными шины MicroLAN.После завершения преобразования полученное значение сравнивается с величинами, хранящимися в регистрах TH и TL. Если измеренная температура выходит за установленные пределы, устанавливается сигнальный “флаг” (впрочем, его установка производится после каждого измерения). Выходные температурные данные DS18B20 калиброваны в градусах Цельсия.Внешний вид датчика в разных, корпусных исполнениях

Расположение выводов

 1- GND – корпус.2- DQ – линия вводавывода данных.3- Vdd – питание датчика.

Схема подключения

На один датчик
Программная часть. Подключение одного датчика

Подключим единственный датчик к микроконтроллеру atmega8535. Теперь начинаем творить программную часть. Подключаем к основному коду программы заголовочный файл ds18b20.h

#include Чтобы начать работать с одним датчиком, достаточно воспользоваться двумя функциями:

Функция инициализации

ds18b20_init( адрес датчика, нижний порог Т, верхний порог Т, разрешающая способность );

функция чтения температуры

ds18b20_temperature( адрес датчика );

Структура будущей программы

……

void main( void )

{   …  … 

  if( //инициализация ds18b20 ) //Проверяем, есть ли устройство на линии 1-wire

  {     

    while( 1 ){ //Цикл, в котором выполняется чтение и вывод значения температуры на LCD

   

    if( ) { } //Проверяем считанную температуру на отрицательное значение

   

    if( ) { } //Аварийное состояние считанной температуры

       }

  }else{ } //Ошибка – устройство не найденно 

}

Полный код программы:

//Цель: производим чтение температуры и выводим на LCD индикатор
#include
#include
#include

#include
#asm  .equ __lcd_port=0x15; PORTC#endasm
#include
#asm    .equ __w1_port=0x1B; PORTA  .equ __w1_bit=0 

#endasm

int temper;                //Переменая для хранения значения температуры

unsigned char lcdBuff[16]; //Массив для хранения фоматированной строки

#pragma rl+

char *str1=”найден датчик ds18b20″;
char *str2=”датчик ds18b20 отсутствует”;
#pragma rl-

void main( void )

{  lcd_init( 16 );  lcd_clear( );   

  if( ds18b20_init( 0, 30, 60, DS18B20_12BIT_RES ) ) //инициализация датчика. Анализ присутствия датчика

  {    lcd_clear( ); lcd_puts( str1 ); delay_ms( 1000 );     

  while( 1 ){

   

    temper=ds18b20_temperature( 0 ); //Чтение температуры, адрес нуль

    delay_ms( 30 );   

  //Анализ на отрицательную температуру

  if( temper>1000 ) { temper=4096-temper; temper=-temper; }

//если переменая temper больше 1000                                                            //То вычитаем от 4096 значение                                                            //переменой temper и далее

                                                            //присваиваем ей знак минус

if( temper>60 ) { /* какая-либо функция, например alarm ( ); */ } //Аварийное состояние считанной температуры
  sprintf( lcdBuff,”t %uxdfC”, temper ); //помещаем форматированую строку в массив
   //выводим на индикатор значение температуры  lcd_clear( ); lcd_gotoxy( 0,0 ); lcd_puts( lcdBuff ); }    

  }else{ lcd_clear( ); lcd_puts( str2 ); }  

}

Проект использования одного датчика ds18b20

Если необходимо выводить значение температуры с точностью до одной сотой, необходимо в функции sprintf заменить тип формата %u на %.3f ( как использовать этот тип формата, будет описано ниже ) и переменную temper объявить как float.


Описание программной части
Функция инициализации датчика ds18b20:

unsigned char ds18b20_init( unsigned char *addr, signed char temp_low, signed char temp_high, unsigned char resolution );

*addr            – адрес датчика ( ROM данные ), к которому обращается микроконтроллер в случае, когда их много.

                         Если датчик один, то всегда нуль.  

temp_low      – значение нижнего порога температуры

temp_high    – значение верхнего порога температуры
resolution  – установка разрешающей способности цифрового преобразователя температуры к 9, 10, 11, или 12                        битам. Принимает значения:  • DS18B20_9BIT_RES• DS18B20_10BIT_RES• DS18B20_11BIT_RES

• DS18B20_12BIT_RES

Функция возвращает положительное значение в случае присутствия датчика на линии 1-wire.Функция чтения температуры:

float ds18b20_temperature( unsigned char *addr );

*addr – адрес датчика, к которому обращается микроконтроллер в случае, когда их много. Если датчик один, то всегда нуль.

Функция возвращает число с плавающей точкой значения температуры ( тип float

Подключение двух датчиков

При подключении более одного датчика ds18b20 на линию 1-wire, необходимо выполнять чтение ROM каждого датчика, чтоб обращаться к каждому индивидуально. Используя функцию w1_search, производим поиск устройств 1-wire, тем самым выясняем их количество на линии и производим чтение ROM данных каждого датчика:

unsigned char devices;
unsigned char rom_code[2][9];

devices=w1_search( 0xF0, rom_code ); //0xF0 можно заменить на DS18B20_SEARCH_ROM_CMD

 
Переменная devices будет содержать целое число найденых устройств 1-wire ( датчики ds18b20 ), а массив rom_code будет содержать 8байт ( 64бита ) уникального кода датчика ds18b20. Этот код имеет следующий состав:Теперь по прочитанному ROM-коду можно обращаться к соответствующим датчикам.

Полный код программы

#include
#include
#include
#include
#asm  .equ __lcd_port=0x15; PORTC

#endasm

#include
#asm    .equ __w1_port=0x1B; PORTA  .equ __w1_bit=0 

#endasm 

#pragma rl+

char*str1=”ЧТЕНИЕ ТЕМПЕРАТУРЫ”;
char *str2=”No SENSOR!”;
#pragma rl-

unsigned char devices;

unsigned char LcdBuffDevices[20];
unsigned char RomCode[2][9];
unsigned char LcdBuff1[20];
unsigned char LcdBuff2[20];

void main( void )

{  lcd_init( 20 );  lcd_clear( );   

  devices=w1_search( DS18B20_SEARCH_ROM_CMD, RomCode ); //поиск датчиков на линии 1-wire

   

  if( devices )

  {

    sprintf( LcdBuffDevices,”DS18B20 = %u”, devices ); //выводим информацию о кол-ве датчиков

       lcd_gotoxy( 0,1 ); lcd_puts( LcdBuffDevices ); lcd_gotoxy( 0,0 ); lcd_puts( str1 );   

    ds18b20_init( &RomCode[0][0], 30, 60, DS18B20_12BIT_RES ); //инициализация первого датчика

    ds18b20_init( &RomCode[1][0], 30, 60, DS18B20_12BIT_RES ); //инициализация второго датчика

  while( 1 )

  {

    //чтение температуры первого датчика

    sprintf( LcdBuff1,”t1 %.2f xefC”, ds18b20_temperature( &RomCode[0][0] ) );
     lcd_gotoxy( 0,2 ); lcd_puts( LcdBuff1 );   

    //чтение температуры второго датчика

    sprintf( LcdBuff2,”t2 %.2f xefC”, ds18b20_temperature( &RomCode[1][0] ) );Впрочем инициализацию можно не выполнять, в том случае, если от датчика не требуется чтение «аварии», тем более по умолчанию установлено 12 битное значение преобразования температуры. 

Вспомним функцию sprintf ( Функция sprintf стандартной библиотеки stdio.h ).Используем тип формата f – число с плавающей точкой в записи с фиксированной десятичной точкой, т.е:

%.0f – без вывода после запятой   t 20 'C
%.1f – единицы                            t 20.6 'C
%.2f – десятки                            t 20.62 'C
%.3f – сотки                               t 20.625 'C

Чтобы данный тип формата работал в функции, необходимо в среде CodeVisionAVR зайти в Configure the project выбрать вкладку C Compiler, Code Generation выбрать из списка (s)printf Features строку float, width, precision 

Цикл программы, в котором происходит чтение температуры датчиковwhile( 1 )  {

    //чтение температуры первого датчика

    sprintf( LcdBuff1,”t1 %.2f xefC”, ds18b20_temperature( &RomCode[0][0] ) );    lcd_gotoxy( 0,2 ); lcd_puts( LcdBuff1 );   

    //чтение температуры второго датчика

    sprintf( LcdBuff2,”t2 %.2f xefC”, ds18b20_temperature( &RomCode[1][0] ) );    lcd_gotoxy( 0,3 ); lcd_puts( LcdBuff2 );

  }

Можно изменить на такой, более удобный

unsigned char num=1;

while( 1 )  {

    //чтение температуры датчиков (  испытание на два датчика )

    do{ sprintf( LcdBuff1,”t%u %.2f xefC”,num, ds18b20_temperature( &RomCode[j][0] ) );        lcd_gotoxy( 0,j );    lcd_puts( lcdBuff );   

    if(

num