Форум программистов, компьютерный форум, киберфорум
Наши страницы
Микроконтроллеры Atmega AVR
Войти
Регистрация
Восстановить пароль
 
Рейтинг 4.50/4: Рейтинг темы: голосов - 4, средняя оценка - 4.50
Krendelyok
0 / 0 / 0
Регистрация: 14.09.2016
Сообщений: 2
1

AtTiny13 + DHT11 + TM1637. Странное поведение

12.11.2016, 08:30. Просмотров 831. Ответов 1
Метки нет (Все метки)

Написал код для работы AtTiny13, DHT11, TM1637. Всё удачно уместилось в памяти микрухи. На первый взгляд всё работает. Данные передаются без сбоев, контрольная сумма в норме. Вот видео , если кому интересно. Но обнаружилась проблема. Периодически, примерно каждые 5-6 запросов данных от датчика DHT11, показания влажности и температуры скачут. На 5 - 10 единиц могут прыгнуть. При этом рассогласования с контрольной суммой нет. Поначалу списал всё на "несерьезность" датчика, но... Залил тот же самый код в Ардуино Уно - нет никаких скачков. Цепляю датчик и дисплей обратно к тиньке - пляшут данные. Подключаю питание непосредственно от этой ардуинки - не помогает, скачут показания. Припаиваю непосредственно к датчику конденсатор по питанию 100 нФ, как рекомендуется в даташите, - бесполезно. Меняю на тиньке частоты: 1.2, 4.8, 9,6 Мгц - никакого результата. Кстати, по ходу дела, заметил еще одну странность: на 9.6 Мгц тинька в этой связке запускается только от 3 вольт, от пяти - не хочет. На дисплей при этом нужно подавать 5 вольт. В общем, у меня закончились предположения, пришел за помощью к вам. В чем подвох может быть, ведь на ардуинке с тем же питанием показания стабильны?
Код:
Кликните здесь для просмотра всего текста
C
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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
#define F_CPU 1200000UL 
#include <avr/io.h> 
#include <util/delay.h> 
#include <avr/interrupt.h> 
#include <avr/wdt.h> 
 
unsigned char word_dht = 0x00; 
unsigned char time_bit[40]; 
unsigned char data_dht[5]; 
unsigned char bit_dht; 
 
 
#define Clkpin 3 //Clkpin TM1637 
#define Datapin 2 // Datapin TM1637 
#define PIN_DHT 0 //Пин датчика DHT 
 
#define DispPort PORTB // обзываем порт 
 
#define  BRIGHT_LCD 7  //Яркость дисплея от 0 до 7 
       
      unsigned char _PointFlag;    //_PointFlag=1:the clock point on 
      unsigned char _DispType; 
      unsigned char DecPoint; 
      unsigned char BlankingFlag; 
    
       
static unsigned char TubeTab[10] = {0x3f,0x06,0x5b,0x4f,// знакогенератор 
                           0x66,0x6d,0x7d,0x07, 
                           0x7f,0x6f};//,0x77,0x7c, 
                           //0x39,0x5e,0x79,0x71, 
                          // 0x40,0x00};//0~9,A,b,C,d,E,F,"-"," "  
 
void TM1637_writeByte(char wr_data)// служебная функция записи данных по протоколу I2C, с подтверждением (ACK) 
{ 
  unsigned char i; 
    for(i=0;i<8;i++)        
  { 
   DispPort &= ~(1<<Clkpin); 
    if(wr_data & 0x01) 
   { DispPort |= 1<<Datapin;} 
    else {DispPort &= ~(1<<Datapin);} 
   _delay_us(3); 
    wr_data = wr_data>>1;      
    DispPort |= 1<<Clkpin; 
   _delay_us(3);  
  }  
  
  DispPort &= ~(1<<Clkpin); 
  _delay_us(5); 
  DDRB &= ~(1<<Datapin);// если поменяете порт на какой-то другой кроме DispPort, то тут тоже все DDRB на другие DDRx менять надо будет 
  while((PINB & (1<<Datapin))); 
  DDRB |= (1<<Datapin); 
  DispPort |= 1<<Clkpin; 
  _delay_us(2); 
  DispPort &= ~(1<<Clkpin);  
} 
 
void TM1637_start(void) // просто функция "старт" для протокола I2C 
{ 
   DispPort |= 1<<Clkpin; 
   DispPort |= 1<<Datapin; 
 _delay_us(2); 
  DispPort &= ~(1<<Datapin); 
} 
 
void TM1637_stop(void) // просто функция "стоп" для протокола I2C 
{ 
  DispPort &= ~(1<<Clkpin); 
 _delay_us(2); 
  DispPort &= ~(1<<Datapin); 
_delay_us(2); 
  DispPort |= 1<<Clkpin;; 
_delay_us(2); 
  DispPort |= 1<<Datapin; 
} 
 
 
void TM1637_init()// Инициализируем дисплей. Как оказалось, можно удалить из этого блока стандартной библиотеки практически всё, кроме инициализации пинов. 
{ 
   DDRB |= (1<<Clkpin) | (1<<Datapin); 
} 
 
//--------------------------------------------------------------------- 
//Здесь код для датчика DHT11. Писал полностью сам. 
 
void start_tmr_us() 
{ 
TCNT0 = 0x00; 
TCCR0B = 0x01; //Запускаем таймер0 с частотой 1,2 Мгц 
} 
 
void start_dht()                //Отправляем приветствие датчику 
{ 
DDRB |= (1 << PIN_DHT); 
DispPort |= (1 << PIN_DHT); 
DispPort &= ~(1 << PIN_DHT); 
_delay_ms(18); 
DDRB &= ~(1 << PIN_DHT); 
DispPort |= (1 << PIN_DHT); 
while(PINB & (1 << PIN_DHT)){} 
while(~PINB & (1 << PIN_DHT)){} 
while(PINB & (1 << PIN_DHT)){} 
 
} 
 
void get_data()                            //Получаем 40 бит от датчика и записываем в массив 
{ 
for(bit_dht = 0; bit_dht < 40; bit_dht++) 
{ 
while(~PINB & (1 << PIN_DHT)){} 
start_tmr_us(); 
while(PINB & (1 << PIN_DHT)){} 
time_bit[bit_dht] = TCNT0; 
} 
} 
 
void decoder_bit(){ 
unsigned char bit_number = 1; 
unsigned char word_number = 0; 
for(bit_dht = 0; bit_dht < 40; bit_dht++){  //дербаним наш массив из 40 бит (5 байтов), в котором записана длительность каждого полученного  бита 
 
if(time_bit[bit_dht] > 50)                //Если длительность бита больше 28 мкс, 
{ 
word_dht = word_dht << 1;            //сдвигаем байт влево и записываем в младший разряд 1 
word_dht |= (1 << 0); 
}else 
{ 
word_dht = word_dht << 1;        //Если меньше 28 мкс, сдвигаем влево байт и записываем 0 
word_dht &= ~(1 << 0); 
} 
 
if(bit_number == 8){                     //Если имеем дело с восьмым битом, записываем полученный байт в массив 
data_dht[word_number] = word_dht; 
word_number += 1; 
bit_number = 0;        
} 
bit_number +=1; 
} 
} 
 
 
 
ISR (WDT_vect){ 
TM1637_writeByte(0x40);    //Выводим на дисплей Err1 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0xC0); 
TM1637_writeByte(0x79); 
TM1637_writeByte(0x50); 
TM1637_writeByte(0x50); 
TM1637_writeByte(0x06); 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0x8F); 
TM1637_stop(); 
} 
 
 
 
 
int main(void) 
{ 
wdt_reset();            //Запускаем вотч дог 
wdt_enable(WDTO_4S); 
WDTCR |= (1 << WDTIE); 
sei(); 
TM1637_init(); 
_delay_ms(1500);  //Зачем-то тупим немного. Иначе, вроде как, дисплей может не зажечься. 
  
while(1) 
{ 
start_dht();                          
get_data(); 
decoder_bit(); 
if((data_dht[0] + data_dht[1] + data_dht[2] + data_dht[3]) != data_dht[4]) 
{ 
   TM1637_writeByte(0x40);    //Выводим на дисплей Err0 
   TM1637_stop(); 
   TM1637_start(); 
   TM1637_writeByte(0xC0); 
   TM1637_writeByte(0x79); 
   TM1637_writeByte(0x50); 
   TM1637_writeByte(0x50); 
   TM1637_writeByte(0x3f); 
   TM1637_stop(); 
   TM1637_start(); 
   TM1637_writeByte(0x8F); 
   TM1637_stop(); 
   _delay_ms(3500); 
   wdt_reset(); 
   _delay_ms(3500); 
   wdt_reset();    
} 
 
TM1637_start();                     //Выводим на дисплей влажность.  
TM1637_writeByte(0x40); 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0xC0); 
TM1637_writeByte(0x76); 
TM1637_writeByte(0x00); 
TM1637_writeByte(TubeTab[(data_dht[0] / 10) % 10]); 
TM1637_writeByte(TubeTab[data_dht[0] % 10]); 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0x8F); 
TM1637_stop(); 
_delay_ms(2000); 
wdt_reset(); 
 
TM1637_start();                                    //Выводим на дисплей температуру 
TM1637_writeByte(0x40); 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0xC0); 
TM1637_writeByte(TubeTab[(data_dht[2] / 10) % 10]); 
TM1637_writeByte(TubeTab[data_dht[2] % 10]); 
TM1637_writeByte(0x63); 
TM1637_writeByte(0x39); 
TM1637_stop(); 
TM1637_start(); 
TM1637_writeByte(0x8F); 
TM1637_stop(); 
_delay_ms(2000); 
wdt_reset(); 
} 
}
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
12.11.2016, 08:30
Ответы с готовыми решениями:

АЦП Attiny13 странное поведение
Всем привет! Ребята помогите, проблема с АЦП. В конце текста описана. Вот...

Странное поведение ветвления if
Доброго времени суток! Пишу простейшую программу часов на ATtiny2313. Сейчас...

Странное поведение контроллера
Всем привет! Уже который день не можем разобраться с проблемой, очень странного...

Странное поведение текствого ЖК дисплея
Добрый день. Написал собственную библиотечку для работы с AVR жк дисплеем и...

Странное поведение таймера Т2 AtMega8
Всем доброго дня. Я, смешно сказать, пытаюсь помигать светодиодом. С точной...

1
radioegor146
24 / 24 / 16
Регистрация: 14.11.2013
Сообщений: 103
19.12.2016, 01:32 2
Резистор 1К подвешен?
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
19.12.2016, 01:32

Странное поведение таймера с предделителем и без
Всем привет! Atmega64, тактируется внешним кварцем на 16 MHz. На одном из...

Очень странное поведение TWI в Atmega128A.
Итак, установка: Atmega128A с подключенным к ней BMP180. Фьюз-бит совместимости...

Странное поведение 8 меги после ресета
Здравствуйте. Вкуриваю обработчик энкодера, вроде что-то получилось, хотя и не...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2018, vBulletin Solutions, Inc.
Рейтинг@Mail.ru