http://habrahabr.ru/post/241728/
Привет. Недавно пересматривал один из моих любимых фильмов, а именно «Трасса 60» с Эми Смарт в главной роли. Там у главного героя была такая вещица, под названием «шар судьбы», который отвечал ему на разные вопросы. Ну и после просмотра у меня появилась мысль, а почему бы не сделать что-то подобное на микроконтроллере, тем более недавно разблокировал 4 штуки, ATtiny13 которые, когда-то заблокировал по незнанию, что такое фьюзы и с чем его едят.
Размеры устройства можете оценить на фоне моей ладони, получился такой себе высокотехнологичный брелок для ключей.
В конце обязательно добавлю не только схему, файлы для Proteus 7, исходники но и фьюзы, hex-файл, чтобы каждый, кто умеет пользоваться программатором, мог повторить данное устройство.
Код для ATtiny13 написан в Arduino IDE, но в силу большого потребления ресурсов этой IDE, которые к стати итак сильно ограничены в ATtiny13(а именно 1024 байта под код), памяти мне не хватило, и тут мне пригодились мои начальные знания регистров микроконтроллера и мизерный опыт работы с ними, вот что в итоге получилось:
Посмотреть код#define F_CPU 1200000UL // Частота МК в герцах
#include <avr/io.h>
#include <avr/sleep.h> // здесь описаны режимы сна
#include <util/delay.h>
#define led_Yes 0 // grn
#define led_No 1 // red
#define rand_gen 3
#define wait 5000 // тайм аут перехода в спящий режим
void setup() {
//pinMode(led_Yes, OUTPUT);
DDRB |= (1<<led_Yes);
//pinMode(led_No, OUTPUT);
DDRB |= (1<<led_No);
}
void loop() {
randomSeed(analogRead(rand_gen)); // не псевдо радном
byte randomValue;
randomValue = random(0,2); // диапазон генератора случайных чисел от 0 до 1
if(randomValue > 0){
//digitalWrite(led_Yes, HIGH);
PORTB |= (1<<led_Yes);
}
else{
//digitalWrite(led_No, HIGH);
PORTB |= (1<<led_No);
}
_delay_ms(wait);
system_sleep();
}
void system_sleep(){
//digitalWrite(led_No, LOW);
PORTB &= ~(1<<led_No);
//digitalWrite(led_Yes, LOW);
PORTB &= ~(1<<led_Yes);
ADCSRA &= ~(1 << ADEN); // перед сном отключим АЦП
// для уменьшения энергопотребления во сне
set_sleep_mode(SLEEP_MODE_PWR_DOWN); // если спать - то на полную
while(1) {
sleep_enable(); // разрешаем сон
sleep_cpu(); // спать!
}
}
//Размер скетча в двоичном коде: 906 байт (из 1 024 байт максимум)
Как это работает? Дело в том, что мы в строчках
randomSeed(analogRead(rand_gen)); задаём некое значение которое поступает в микроконтроллер из вне, так как порт микроконтроллера ни к чему не подключен, то есть «висит в воздухе», то на нём присутствует некий
белый шум. Для микроконтроллера он выглядит примерно так:
Если микроконтроллер будет выводить не очень ожидаемый ответ, то можно переназначить пины, то есть поменять местами, порты которые заданы при помощи директивы
#define led_Yes 0 и
#define led_No 1 или же ещё простой способ — банально поменять местами светодиоды.
В строчке
#define wait 5000 задаётся тайм аут перехода в спящий режим, то есть, сколько будет светить светодиод.
Посмотрим потребление микроконтроллера в спящем режиме в даташите:
Как видите 0.2 мкА, при таком токе аккумулятор сам по себе быстрее «сядет» чем микроконтроллер его разрядит хотя бы на 1 процент.
В данном коде используется не псевдо генератор случайных чисел(постоянно одна и та же последовательность чисел), этого удалось добиться при помощи функции
randomSeed().
Схема очень простая:
Всего 6 компонентов не считая литиевой батарейки CR2025. Резисторы R1 и R3 добавил чисто для приличия как говорится, без них схема будет работать не хуже чем с ними, правда резистор R1 немного уменьшает ток светодиодов, но нужен, если напряжение питания схемы превышало бы 3 вольта.
Интересно, на сколько нажатий хватит батарейки CR2025, емкость которой к стати примерно 150 мА/ч?
При единичном нажатии на кнопку светодиод горит 5 сек, то есть 60 сек / 5 = 12 нажатий на минуту * 60 минут = 720 непрерывных нажатий в час, за час схема будет потреблять ток 1.5 мА(измерил мультиметром потребление когда горит красный светодиод), при емкости в 150 мА/ч 150/1.5 мА/ч схема проработает 100 часов, так как за час можно совершить максимум 720 нажатий то 720 * 100 = 72 000.
Из грубого подсчёта выходит, что до полного истощения литиевой батарейки CR2025 нужно совершить 72 тысячи нажатий и если клацать кнопку непрерывно, на это уйдёт 100 часов, чуть больше 4-х суток.
Ссылки:
Фильм Трасса 60(википедия);
Даташит на ATtiny13;
Прошивка и программирование ATtiny13 при помощи Arduino;
Arduino IDE;
Функция randomSeed();
Белый шум;
Как экономить место на микроконтроллере?;
Софт которым сделал скриншонт шума — Serial oscilloscope;
Все файлы по этому проекту.
P.S. схема в Proteus работать откажется, будет постоянно гореть один и тот же светодиод, так как данная программа не умеет эмулировать те наводки на порту АЦП, что будут в реальной схеме.