среда, 26 ноября 2014 г.

Суровая правда жизни

#С Юрой пообщался
Я могучий, как джин – все на свете могу!
Строить замки, сносить города!
Но задач по плечу не найти… и грущу…
И картошку копаю тогда.


#Просто так
А снежинки ажурные с неба спускаются
И тихонько ложатся под ноги они
С тихим звоном о землю они разбиваются
Заполняя собою все зимние дни…

Ёжики

А чумазые ежики ходят по комнатам
В одинаковых синих трусах
И грызут эти ежики булки пшеничные
И варенье у них на усах.

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

вторник, 25 ноября 2014 г.

Arduino. Собираем своего робота из компектующих

Ранее упоминал о приобретенном на dx.com наборе для сборки робота. В какой-то момент число замененных комплектующих стало существенным и решил уточнить, во что обойдется аналогичный робот, приобретенный врассыпуху, а не готовым комплектом.

Полный набор сейчас стоит 86.6$.

Если брать все по отдельности, то получим:

КомплектующиеЦена
Платформа с двигателями 26$
4-канальный контроллер двигателей 3.54$
Аналог Arduino Uno R3 8.82$
Сервопривод и ультразвуковой датчик 5.9$
Модуль Bluetooth 4.68$
Стекируемые разъемы 1.98$
Набор соединительных проводов 1.58$
Итого - 52,5$.

Разница получается 34.1$. Но есть нюанс - предлагаемый мной контроллер для управления двигателями одевается СВЕРХУ на ардуинку и закрывает все пины, даже те, которые ему не нужны ля работы. Поэтому надо приложить немного руки и перепаять контактные площадки с дефолтовых на предлагаемые мной в списке (последняя позиция).

Паяльник с регулируемой температурой нагрева жала, подставкой и отсосом для припоя можно приобрести на том же aliexpress.com за 12$. Те, кому лень паять, могут сразу купить вместо клона Arduino Uno R3 - клон Arduino Mega 2560 R3, которая имеет значительно больше свободных портов и которая стоит около 16$ - мы все равно выигрываем около 20$. :о)

Толковый мануал по сборке можно найти на сайте http://chingachgook.net

Управлять роботом можно по Bluetooth с компа или с телефона.

Вот и все. Удачи в разработке скетчей! Да пребудет с вами Сила! ;о)


пятница, 23 мая 2014 г.

Травма сустава. Чем мазать?

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

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

В одном из ящиков я обнаружил большое количество упаковок с различными мазями, из которых только название "Фастум гель" оказалось знакомым благодаря рекламе.

Взял тюбик и под напутственные слова доктора отправился в каюту обрабатывать поврежденную конечность. Намазал. Никакого эффекта. Ну, думаю, может пока впитается... Прошел час, в течении которого довольно много бегал по рабочим помещениям. Мази как будто и не было, а плечо болит - ни обезбаливающего, ни согревающего эффекта не наблюдается. Смыл остатки мази и снова зашел к доктору. На этот раз взял мазь "Капсикам".

На счет обезбаливающего - не знаю, но греть стало почти сразу. Особенно хорошо стало припекать когда разогрелся немного. УХ! Хорошо! :о)

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

Вернул доктору мазь, обсудили лечебный эффект от использованных препаратов. Доктору тоже "Фастум гель" не очень нравится. С "Капсикамом" он лично не сталкивался, а по опыту использования очень хвалил "Кетонал".

Резюмируя. "Фастум гель" - ерунда, не достойная чести занимать место в аптечке приличного спортсмена. "Капсикам" - прошел полевые испытания и показал неплохие результаты. "Кетонал" - то, что доктор прописал.

P.S. Может возникнуть вопрос, почему доктор изначально не предложил "Кетанол". А за время, пока я лечился у нас успели врачи поменяться. :о)


вторник, 20 мая 2014 г.

Arduino. Работаем с часами DS3231 по i2C через Wire.

Копнул более основательно часики.

Для начала запустил сканер устройств на шине i2C. Рекомендую сохранить этот скетч в свою библиотеку, чтобы он был под руками - очень полезная штука!

Сканер обнаруживает два устройства на шине i2C по следующим адресам:
1. 0x57 - это адрес микросхемы памяти AT24C32
2. 0x68 - адрес микросхемы DS3231SN

Cтоит учитывать - сканер возвращает адреса в формате шестнадцатеричной системы. Т.е в привычном нам десятичном формате получим адреса 87 и 104. Нас интересует адрес 0x68 (104). Сделал себе первую пометочку.

Следующий шаг - скачать даташит к микросхеме. Найти его можно, например, по этой ссылке
Читаю, радуюсь большому количеству английских букв и разнообразных таблиц. :)

Самое интересное началось на странице 11, где присутствует следующая таблица:

Figure 1. Timekeeing Registers

Addressbit 7
MSB
bit 6bit 5bit 4bit 3bit 2bit 1bit 0
LSB
FunctionRange
00H010 secondssecondsSeconds00-59
01H010 minutesminutesMinutes00-59
02H012/24AM/PM
10 hour
10 hourhourHours1-12+AM/PM
00-23
03H00000dayDay1-7
04H0010 datedateDate00-31
05HCentury0010 monthmonthMonth/Century01-12+Century
06H10 yearyearYear00-99
07HA1M110 secondssecondsAlarm 1
seconds
00-59
08HA1M210 minutesminutesAlarm 1
minutes
00-59
09HA1M312/24AM/PM
10 hour
10 hourhourAlarm 1
hours
1-12+AM/PM
00-23
0AHA1M4DY/DT10 datedayAlarm 1 day1-7
dateAlarm 1 date1-31
0BHA2M210 minutesminutesAlarm 2 minutes00-59
0CHA2M312/24AM/PM
10 hour
10 hourhourAlarm 2 hours1-12+AM/PM
00-23
0DHA2M4DY/DT10 datedayAlarm 2 day1-7
dateAlarm 2 date1-31
0EHEOSCBBSQWCONVRS2RS1INTCNA2IEA1IEControl---
0FHOSF000EN32kHzBSYA2FA1FControl/status---
10HSIGNDATADATADATADATADATADATADATAAgign Offset---
11HSIGNDATADATADATADATADATADATADATAMSB of Temp---
12HDATADATA000000LSB of Temp---
Рекомендую все-таки скачать даташит - там есть такие замечательные отметочки в оригинальной таблице - крышечки. Как я понял, обозначают, какое состояние бита отвечает за состояние AM/PM или DY/DT, например. В моей таблице этого нет.

Первоначально интерес тут представляют строки с 00H до 06H и c 0EH до 12H. Блок с алармами (07H-0DH) пока пропущу - им займусь позже.

Далее приведен скетч, работающий напрямую с данными на DS3231.

#include <Wire.h>
 #define DS3231_I2C_ADDRESS 0x68
  
 byte seconds, minutes, hours, day, date, month, year;
 char weekDay[4];  
 byte tMSB, tLSB;
 float temp3231;
  
 void setup()
 {
   Wire.begin();
   Serial.begin(9600);
   //set control register to output square wave on pin 3 at 1Hz
   Wire.beginTransmission(DS3231_I2C_ADDRESS); // 104 is DS3231 device address
   Wire.write(0x0E); //выставляемся в 14й байт 
   Wire.write(B00000000); //сбрасываем контрольные регистры 
   Wire.write(B10001000); //выставляем 1 на статус OSF и En32kHz
   Wire.endTransmission();
 }
  
 void loop() {    
   watchConsole(); //читаем консоль. Если нажата "T" читаем из консоли следующий 7 байт и устанавливаем время.    
   get3231Date(); //получаем данные 

          //и потом их выводим через Serial.print()    
   Serial.print(weekDay); Serial.print(", "); Serial.print(month, DEC); Serial.print("/"); Serial.print(date, DEC); Serial.print("/"); Serial.print(year, DEC); Serial.print(" - ");
   Serial.print(hours, DEC); Serial.print(":"); Serial.print(minutes, DEC); Serial.print(":"); Serial.print(seconds, DEC);    
   Serial.print("   Temperature: "); Serial.print(get3231Temp());Serial.print("; ");
          Serial.println(get3231Register(0x0F)); //получаем и выводим статусовый регистр
   delay(1000);
 }
  
 // Convert normal decimal numbers to binary coded decimal
 byte decToBcd(byte val)
 {
   return ( (val/10*16) + (val%10) );
 }
  
 void watchConsole()
 {
   if (Serial.available()) {      // Look for char in serial queue and process if found
     if (Serial.read() == 84) {      //If command = "T" Set Date
       set3231Date();
       get3231Date();
       Serial.println(" ");
     }
   }
 }
    
 void set3231Date()
 {
 //T(sec)(min)(hour)(dayOfWeek)(dayOfMonth)(month)(year)
 //T(00-59)(00-59)(00-23)(1-7)(01-31)(01-12)(00-99)
 //Example: 02-Feb-09 @ 19:57:11 for the 3rd day of the week -> T1157193020209
  
   seconds = (byte) ((Serial.read() - 48) * 10 + (Serial.read() - 48)); // Use of (byte) type casting and ascii math to achieve result. 
   minutes = (byte) ((Serial.read() - 48) *10 +  (Serial.read() - 48));
   hours   = (byte) ((Serial.read() - 48) *10 +  (Serial.read() - 48));
   day     = (byte) (Serial.read() - 48);
   date    = (byte) ((Serial.read() - 48) *10 +  (Serial.read() - 48));
   month   = (byte) ((Serial.read() - 48) *10 +  (Serial.read() - 48));
   year    = (byte) ((Serial.read() - 48) *10 +  (Serial.read() - 48));

   Wire.beginTransmission(DS3231_I2C_ADDRESS); //начали сессию
   Wire.write(0x00); //поставляемся в нулевой байт для чтения данных
   Wire.write(decToBcd(seconds));
   Wire.write(decToBcd(minutes));
   Wire.write(decToBcd(hours));
   Wire.write(decToBcd(day));
   Wire.write(decToBcd(date));
   Wire.write(decToBcd(month));
   Wire.write(decToBcd(year));
   Wire.endTransmission();//закрыли сессию
 }
  
 byte get3231Register(byte regNo) {
   // send request to receive data starting at register regNo
   Wire.beginTransmission(DS3231_I2C_ADDRESS); // 104 is DS3231 device address
   Wire.write(regNo); // start at register regNo
   Wire.endTransmission();
   Wire.requestFrom(DS3231_I2C_ADDRESS, 1); // request one byte
   if(Wire.available()) return  Wire.read();
 }
  
 void set3231Register(byte regNo, byte value) {
   Wire.beginTransmission(DS3231_I2C_ADDRESS);
   Wire.write(regNo);
   Wire.write(value);
   Wire.endTransmission();
 }
  
 void get3231Date()
 {
   // send request to receive data starting at register 0
   Wire.beginTransmission(DS3231_I2C_ADDRESS); // 104 is DS3231 device address
   Wire.write(0x00); // start at register 0
   Wire.endTransmission();
   Wire.requestFrom(DS3231_I2C_ADDRESS, 7); // request seven bytes
  
   if(Wire.available()) {
     seconds = Wire.read(); // get seconds
     minutes = Wire.read(); // get minutes
     hours   = Wire.read();   // get hours
     day     = Wire.read();
     date    = Wire.read();
     month   = Wire.read(); //temp month
     year    = Wire.read();

//Конвертация происходит очень просто - битовым оператором & мы очищаем МЛАДШИЕ (0,1,2,3) четыре бита , сдвигаем все вправо на 4 позиции (очищенные биты
//вылезут при этом с левой стороны и результат умножаем на 10, т.к. в старших битах хранится информация о десятках секунд.
//После этого битовым оператором & очищаем СТАРШИЕ (4,5,6,7)четыре бита и получаем секунды. Затем суммируем десятки секунд и единиц.
     seconds = (((seconds & B11110000)>>4)*10 + (seconds & B00001111));
     minutes = (((minutes & B11110000)>>4)*10 + (minutes & B00001111));
     hours   = (((hours & B00110000)>>4)*10 + (hours & B00001111)); 
     day     = (day & B00000111); // 1-7
     date    = (((date & B00110000)>>4)*10 + (date & B00001111)); // 1-31
     month   = (((month & B00010000)>>4)*10 + (month & B00001111)); //msb7 is century overflow
     year    = (((year & B11110000)>>4)*10 + (year & B00001111));
   }
   else {
     //oh noes, no data!
   }
    
   switch (day) {     
     case 1:
       strcpy(weekDay, "Mon");
       break;
     case 2:
       strcpy(weekDay, "Tue");
       break;
     case 3:
       strcpy(weekDay, "Wed");
       break;
     case 4:
       strcpy(weekDay, "Thu");
       break;
     case 5:
       strcpy(weekDay, "Fri");
       break;
     case 6:
       strcpy(weekDay, "Sat");
       break;
            case 7:
       strcpy(weekDay, "Sun");
       break;
   }
 }
  
 float get3231Temp()
 {
   //temp registers (11h-12h) get updated automatically every 64s
   Wire.beginTransmission(DS3231_I2C_ADDRESS);
   Wire.write(0x11);
   Wire.endTransmission();
   Wire.requestFrom(DS3231_I2C_ADDRESS, 2);
    
   if(Wire.available()) {
     tMSB = Wire.read(); //2's complement int portion
     tLSB = Wire.read(); //fraction portion      
     temp3231 = (tMSB & B01111111); //do 2's math on Tmsb
     temp3231 += ( (tLSB >> 6) * 0.25 ); //only care about bits 7 & 8
   }
   else {
     //oh noes, no data!
   }
      
   return temp3231;
 }

Вот как-то так. Непонятно пока, что произойдет при совпадении текущего времени с установленным в байтах 07h-0Ah и в 0Bh-0Dh, но этим займусь завтра.


понедельник, 19 мая 2014 г.

Arduino. Программирование

Начал разбираться, как использовать аппаратный будильник часов и понял, что познания мои о среде программирования в Arduino весьма поверхностны. Нашел полезный ресурс и решил поделиться. Так же справочник "языка Arduino" можно посмотреть на российском сайте Arduino.

Еще раз напомню о замечательной подборке материалов по Arduino, датчикам и библиотекам на robocraft.ru


воскресенье, 18 мая 2014 г.

Arduino. Звук - пищим динамиком

В коробке остался 8Ом-ный динамик, 0,5W. Порылся в закромах, нашел ссылку на статью о подключении динамика к Arduino.

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

Динамик маломощный, звук тихий. Я подключал без сопротивления 100 Ом-ного. Надо помнить, что это понижающий резистор, и при больших Омах мы просто не услышим свой динамик. Если найти низкоомный потенциометр (менее или равный 1кОм), то можно будет регулировать громкость звучания динамика.

Простейший скетч

void setup() {
}

void loop() {
tone(8, 311,300);
delay(500);
noTone(8);
}
}


суббота, 17 мая 2014 г.

Arduino. HC-SR04 - ультразвуковой модуль измерения расстояния

Имеющихся в наличии датчиков все меньше и меньше и выбора особого уже не остается. :о) Сегодня подключаем датчик HC-SR04 - ультразвуковой датчик измерения расстояния.

Дополнительных библиотек к нему не надо, что существенно упрощает нам жизнь. ;о) Немного дополнительной информации можно почерпнуть отсюда или отсюда.

Схема подключения самая простая.

Скетч для работы с датчиком
#define Trig 9
#define Echo 8

void setup()
{
pinMode(Trig, OUTPUT); //инициируем как выход
pinMode(Echo, INPUT); //инициируем как вход
Serial.begin(9600);
/* задаем скорость общения. В нашем случае с компьютером */
}
unsigned int impulseTime=0;
unsigned int distance_sm=0;

void loop()
{
digitalWrite(Trig, HIGH);
/* Подаем импульс на вход trig дальномера */
delayMicroseconds(10); // равный 10 микросекундам
digitalWrite(Trig, LOW); // Отключаем
impulseTime=pulseIn(Echo, HIGH); // Замеряем длину импульса
distance_sm=impulseTime/58; // Пересчитываем в сантиметры
Serial.println(distance_sm); // Выводим на порт
delay(100);
/* ждем 0.1 секунды, Следующий импульс может быть излучён, только после исчезновения эха от предыдущего. Это время называется периодом цикла (cycle period).
Рекомендованный период между импульсами должен быть не менее 50 мс. */
}

Думаю, как подключить дальномер к УСТРОЙСТВУ и как изменить скетч - понятно. Только стоит обратить внимание на то, что Digital 8 у нас занят датчиком HC-SR501. Поэтому предлагаю Echo подключить к Digital 7, и внести необходимые изменения в блоке определения пинов.

На этом имеющиеся в наличии датчики закончились - остались только два двигателя и контроллер к ним. И еще нашел динамик 8 Ом-ный. Так что впереди борьба с колесами и музыка. :о)

При тестировании выявился такой неприятный момент - на дистанции более 2 метров расстояние фиксируется не стабильно. Т.е. и 201 покажет и 203, но может и 820 показать или 930. Причем чем дальше от приемника, тем реже на экране возникают правильные цифры. До двух метров меня дальномер "ловил" хорошо. Как вариант может попробовать сделать трубку из бумаги на излучатель, чтобы сузить сектор? Должно быть меньше помех и проще будет определять, куда излучение "полетело". Хотя, если все так просто, почему производители изначально не делают трубки более длинными? Непонятно...

Трубка не помогла. На расстоянии до 150 см фиксация четкая. Выше - начинают выскакивать 875 и другие "неправильные" числа. Вывод - датчик стабильно работает на полтора метра. Дальше - сложно будет отсеять достоверные результаты от помех.


пятница, 16 мая 2014 г.

Arduino. Датчик движения HC-SR501

Дело было вечером, делать было нечего... Решил пересмотреть свою копилку материалов по датчикам к Arduino. Наткнулся на схему проверки работоспособности датчика движения HC-SR501

Вместо рекомендуемого 4.5 кОм резистра воткнул 15 кОм (ну, что нашел). Датчик весело замигал светодиодом при движении. Много интересного про датчик можно прочитать, например, на этом ресурсе.

До подключения к плате осталось пара шагов - подключил выход Out к 8 разъему (Digital) на плате микроконтроллера. По совету, увиденному в одном из форумов подвязал Out резистором к земле. Номинал резистора оставил 15 кОм. Питание (+5В) и землю взял с платы.

Простейший скетч для проверки работы датчика с Arduino

int pirPin = 8;
int val;

void setup() {
    pinMode(pirPin,INPUT);
    Serial.begin(9600);
}

void loop() {
    val = digitalRead(pirPin); //read state of the PIR
    if (val == LOW) {
      Serial.println("No motion"); //if the value read is low, there was no motion
    }
    else {
      Serial.println("Motion!"); //if the value read was high, there was motion
    }
    delay(1000);
}

Когда все заработало, перецепил все на схему к уже собранному УСТРОЙСТВУ. Модифицировал скетч - при появлении сигнала от датчика включается подсветка экрана и горит 5 секунд.

#include Wire.h
#include DS3231.h
#include LiquidCrystal.h
/*
The circuit:
* 5V to Arduino 5V pin
* GND to Arduino GND pin
* SDA (data) to Analog #4
* SCL (clock) to Analog #5
*/

// Connect via i2c, default address #0 (A0-A2 not jumpered)
LiquidCrystal lcd(0);
DS3231 Clock;
bool h12=false;//True is 12-h, false is 24-hour.;
bool PM;
bool Century=false;

int pirPin = 8;
int val;
int msec;

void setup() {
// set up the LCD's number of rows and columns:
lcd.begin(16, 2);
lcd.clear();
lcd.setBacklight(LOW);
Wire.begin();
pinMode(pirPin,INPUT);
}

void loop() {
int second,minute,hour,date,month,year,temperature;

val = digitalRead(pirPin); //read state of the PIR
if (val == LOW){
lcd.setBacklight(LOW); //if the value read is low, there was no motion
}
else {
lcd.setBacklight(HIGH);
msec=millis()/1000;
}
if (millis()/1000-msec<5) lcd.setBacklight(HIGH);

year=Clock.getYear();
month=Clock.getMonth(Century);
date=Clock.getDate();
hour=Clock.getHour(h12,PM);
minute=Clock.getMinute();
second=Clock.getSecond();
temperature=Clock.getTemperature();

lcd.print("20");
lcd.print(year,DEC);
lcd.print('-');
if (month<10) lcd.print('0');
lcd.print(month,DEC);
lcd.print('-');
if (date<10) lcd.print('0');
lcd.print(date,DEC);

// set the cursor to column 0, line 1
// (note: line 1 is the second row, since counting begins with 0):
lcd.setCursor(0, 1);
if (hour<10) lcd.print('0');
lcd.print(hour,DEC);
lcd.print(':');
if (minute<10) lcd.print('0');
lcd.print(minute,DEC);
lcd.print(':');
if (second<10) lcd.print('0');
lcd.print(second,DEC);
lcd.print(' ');
lcd.print("Temp:");
lcd.print(temperature);
delay(500);
lcd.setCursor(0,0);
}

тут завтра будет фотография УСТРОЙСТВА


Arduino. DS3231 и I2C/SPI LCD1602 в одном флаконе

Сегодня мне удалось собрать первый функциональный прибор, который может работать без компьютера. Ранее, я уже писал о подключении высокоточных часов и I2C LCD 1602 по шине I2C. Следующей стояла задача совместного использования этих устройств.

В интернете нашел несколько статей по использованию шины I2C. Есть более развернутые статьи, есть менее. Интересующимся рекомендую вот эту статью. Или вот эту. В статье рекомендуют привязывать разъемы SDA и SCL к +5 вольтам. Номинал резисторов от 1кОм до 10 кОм. Я поставил 1кОм (других не нашлось).

Схема сборкиФотография

Скетч просто собрал из двух, использовавшихся ранее для часов и экрана

#include Wire.h
#include DS3231.h
#include LiquidCrystal.h
/**
The circuit:
* 5V to Arduino 5V pin
* GND to Arduino GND pin
* SDA (data) to Analog #4
* SCL (clock) to Analog #5
**/

// Connect via i2c, default address #0 (A0-A2 not jumpered)
LiquidCrystal lcd(0);
DS3231 Clock;
bool h12=false;//True is 12-h, false is 24-hour.;
bool PM;
bool Century=false;

void setup() {
// set up the LCD's number of rows and columns:
lcd.begin(16, 2);
lcd.clear();
lcd.setBacklight(LOW);
Wire.begin();
}

void loop() {
int second,minute,hour,date,month,year,temperature;

year=Clock.getYear();
month=Clock.getMonth(Century);
date=Clock.getDate();
hour=Clock.getHour(h12,PM);
minute=Clock.getMinute();
second=Clock.getSecond();
temperature=Clock.getTemperature();

lcd.print("20");
lcd.print(year,DEC);
lcd.print('-');
if (month<10) lcd.print('0');
lcd.print(month,DEC);
lcd.print('-');
if (date<10) lcd.print('0');
lcd.print(date,DEC);

// set the cursor to column 0, line 1
// (note: line 1 is the second row, since counting begins with 0):
lcd.setCursor(0, 1);
if (hour<10) lcd.print('0');
lcd.print(hour,DEC);
lcd.print(':');
if (minute<10) lcd.print('0');
lcd.print(minute,DEC);
lcd.print(':');
if (second<10) lcd.print('0');
lcd.print(second,DEC);
lcd.print(' ');
lcd.print("Temp:");
lcd.print(temperature);
delay(500);
lcd.setCursor(0,0);
}


Вот такие вот часы у меня получились. ;о)


Arduino. DS3231 - высокоточные часы реального времени

Иду дальше по плану. Высокоточные часы. Тут все было предельно просто - на странице интернет-магазина сразу имелась ссылка на библиотеки. Распаковал, установил файлы из папки DS3231_TEST, запустил скетч из примера... Замечательный девайс! Показывает время и температуру. А наличие батарейки позволяет таймеру работать и без подключения к компьютеру.

Вот упрощенный текст скетча
#include DS3231.h
#include Wire.h

DS3231 Clock;
bool Century=false;
bool h12;
bool PM;
byte year, month, date, DoW, hour, minute, second;

void setup() {
// Start the I2C interface
Wire.begin();
Clock.setSecond(50);//Set the second
Clock.setMinute(59);//Set the minute
Clock.setHour(11); //Set the hour
Clock.setDoW(5); //Set the day of the week
Clock.setDate(31); //Set the date of the month
Clock.setMonth(5); //Set the month of the year
Clock.setYear(13); //Set the year (Last two digits of the year)
// Start the serial interface
Serial.begin(9600);
}

void loop() {
int second,minute,hour,date,month,year,temperature;
second=Clock.getSecond();
minute=Clock.getMinute();
hour=Clock.getHour(h12, PM);
date=Clock.getDate();
month=Clock.getMonth(Century);
year=Clock.getYear();
temperature=Clock.getTemperature();

Serial.print("20");
Serial.print(year,DEC);
Serial.print('-');
Serial.print(month,DEC);
Serial.print('-');
Serial.print(date,DEC);
Serial.print(' ');
Serial.print(hour,DEC);
Serial.print(':');
Serial.print(minute,DEC);
Serial.print(':');
Serial.print(second,DEC);
Serial.print('\n');
Serial.print("Temperature=");
Serial.print(temperature);
Serial.print('\n');
delay(1000);
}

И схема подключения. Аналогична схеме подключения LCD.

Следующий шаг - заставить работать два устройства I2C на одной паре проводов. Т.е. чтобы часы выводили данные на LCD, а не через виртуальный COM-порт в среду разработки. Как разберусь - напишу. ;)


Arduino. I2C/SPI LCD1602 Module

Вчера почти весь день боролся с модулем IIC/I2C/TWI SPI Serial LCD 1602.

И так его уговаривал, и так... В конце концов решил подойти к решению вопроса системно - для начала узнать, какой адрес имеет подключаемый модуль. Скачал скетч сканера I2C, запустил его при подключенном экране. Узнал, что устройство имеет адрес 0х20. Отлично! Казалось бы, вот оно, счастье.. Но знание адреса не помогло. Перепробовал кучу разных библиотек для I2C LCD, и все никак. Менял провода, платы с микроконтроллером... Тоже без толку.

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

В конце концов наткнулся на страничку производителя модуля. И там, помимо схемы устройства оказались еще и библиотеки для модуля. Заработала вот эта библиотека SPI_IIC_LCD library for Arduino 1.0. Причем заработал код из примера, при выставленном адресе не 0х20, а просто 0! Почему так получилось - не понял. Будет время - покопаюсь. Но сейчас модуль исправно выводит надписи, мигает подсветкой и вообще ведет себя образцово.

Вот текст скетча

#include Wire.h
#include LiquidCrystal.h

// Connect via i2c, default address #0 (A0-A2 not jumpered)
LiquidCrystal lcd(0);

void setup() {
// set up the LCD's number of rows and columns:
lcd.begin(16, 2);
lcd.print("hello, world!");
}

void loop() {
// set the cursor to column 0, line 1
// (note: line 1 is the second row, since counting begins with 0):
lcd.setCursor(0, 1);
// print the number of seconds since reset:
lcd.print(millis()/1000);
lcd.setBacklight(HIGH);
delay(500);
lcd.setBacklight(LOW);
delay(500);
}

И схема подключения, на всякий пожарный. :о)


среда, 14 мая 2014 г.

Arduino. Первое включение. Мигаем светодиодом.

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

Когда открывал посылку, был удивлен размерами деталей. Ну, вот как такую фот финтифлюшку брать моими пальцами! :о)

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

IDE качал с официального сайта. Единственно, скачивал релиз Arduino 1.5.6-r2 BETA (with support for Arduino Yún and Arduino Due boards Кстати, развернутая из архива, а не проинсталлированная, среда отлично работает.

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


/* Blink
Turns on an LED on for one second, then off for one second, repeatedly.
This example code is in the public domain.
*/
// Pin 13 has an LED connected on most Arduino boards.
// give it a name:
int led = 13;
// the setup routine runs once when you press reset:
void setup() {
// initialize the digital pin as an output.
pinMode(led, OUTPUT);
}
// the loop routine runs over and over again forever:
void loop() {
digitalWrite(led, HIGH); // turn the LED on (HIGH is the voltage level)
delay(1000); // wait for a second
digitalWrite(led, LOW); // turn the LED off by making the voltage LOW
delay(1000); // wait for a second

digitalWrite(led, HIGH);
delay(500);
digitalWrite(led, LOW);
delay(1000);

digitalWrite(led, HIGH);
delay(100);
digitalWrite(led, LOW);
delay(1000);

}

Теперь светодиод мигает, как и раньше через равные промежутки времени, но время свечения различается. Вот после этого я поверил, что это заработал залитый мной скетч! :о)

Обе платы пока работают идентично. Завтра попробую цеплять сенсоры.


четверг, 3 апреля 2014 г.

Arduino есть. Что дальше?

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

Компонент Описание Функционал
Плата микроконтроллера Собственно, сама по себе плата может только мигать светодиодом и слать на ПК запрограммированные сигналы. Пока не очень интересно. Но перспективы есть - контроллер-то программируемый!.:о)
2,6" ЖК-дисплей Можно будет снимать какие-то данные без подключения к компьютеру. УСТРОЙСТВО начинает обретать автономность. ;о) Кстати, данный экран имеет маркировку 1602, что подсказывает нам о возможности вывода на экран двух строк по 16 символов.
Высокоточные часы реального времени Чтобы было что выводить на экран. Ну и полезный функционал появляется. Так ведь веселее. :о)
Пироэлектрический датчик Теперь УСТРОЙСТВО сможет узнавать, когда к нему кто-то подходит. Зачем? Ну, хотя бы для того, чтобы начать что-то выводить на экран. Предполагается, что в отсутствии зрителей на экран не будет ничего выводиться.
Ультразвуковой модуль измерения расстояния Ну как же без измерителя расстояния! Очень нужный датчик. Пригодится потом, когда УСТРОЙСТВО обзаведется колесами и будет учиться избегать столкновений с препятствиями.
Зуммер У нас есть часы. Почему бы не сделать их будильником? Пусть напоминает о событиях звуковым сигналом. А можно еще и на экран выводить напоминалку. Кроме того видел схемы с подключением 8-омных динамиков. Можно будет попробовать не простые Би-и-и или Дз-з-з реализовывать, а какие-нибудь простенькие мелодии выводить.
Выходной сдвиговый регистр 8 бит - 74HC595 Зачем? Даже себе не смог объяснить. Но так звучит красиво: "Сдвиговый регистр"! В принципе, позволит получить +7 выходных портов, каскадами регистров можно получать еще более значительное увеличение выходных портов. Может быть полезно при управлении большим числом внешних устройств. Например - диодной матрицей.
Входной сдвиговый регистр 8 бит - 74HC165N Вообще без понятия. Но раз уж буду брать выходной регистр, то с входным тоже надо разобраться. До кучи. :о)
10-ти сегментный индикатор А это мой подопытный "кролик", для тестирования 74HC595
Модуль памяти Micro SD Лучше поздно, но задуматься о ведении логов событий. SD-карта - вполне удобный вариант.
Цифровой датчик температуры и влажности Вот представим себе - гуляю я вечером по улице. Подходят ко мне ребята и спрашивают время. Я достаю из кармана УСТРОЙСТВО и говорю им, который час. Ребята слегка офигевают. А если самые ушлые спросят про температуру, я им тоже смогу ответить! Ну, не бить же людей каждый раз за то, что они расстроились, не получив ответ на вопрос о времени! :о)) А если серьезно - это начало блока контроля за растениями. :о) Есть у меня такая задумка на перспективу. Да и в быту, думаю, такая информация может быть не лишней.
Датчик влажности почвы Втыкаем щуп в землю и считываем информацию о влажности грунта. ИМХО, штука полезная. Когда-нибудь можно будет к системе еще и автополив прикрутить, но на первом этапе с лейкой, думаю, я и сам справлюсь. ;о) Датчиков влажности много - указанный тут лишь один из них. Я пока еще предметно вопрос не изучал. Может есть и более качественные. Кстати, большое количество датчиков можно будет прикрутить к ардуине с помощью входного сдвигового регистра. Т.е. контролировать несколько десятков саженцев - не проблема.
Датчик освещенности Штука нужная для подключения дополнительного освещения растениям. Пока это просто памятка о необходимости изучения данного вопроса. Есть фоторезисторы и фотодиоды. Кто-то из них дает более приятные сигналы на выходе, подходящие к моей задаче. Ближе к реализации разберусь и в статье напишу.
Датчик звука Интересно, а можно ли будет управлять УСТРОЙСТВОМ посредством дискретных звуковых сигналов? Разработать набор звуковых кодов (стуков) и использовать для управления. Потом надо будет разобраться с компонентами.
Компас Надеюсь, растения оценят мою заботу о них и начнут активно расти. Вот тогда мне и пригодится компас, чтобы выбираться из зарослей с огорода! :о)) И для навигации УСТРОЙСТВА в будущем пригодится. Вопрос нуждается в дополнительном изучении.
Bluetooth Буду ли по блютусу управлять УСТРОЙСТВОМ - не знаю. Но как вариант, можно использовать датчик для привязки к хозяину. Уровень сигнала падает быстро и используя это, УСТРОЙСТВО сможет определять направление, в котором хозяин от него удаляется. Или оно удаляется от хозяина. :о)

Собственно, на этом немобильная версия УСТРОЙСТВА закончится. К моменту реализации всего вышеуказанного, надеюсь, старшему ребенку надоест его игрушка и удастся задействовать ее в качестве мобильной платформы.

Вот такие вот скромные планы. А еще в планах закончить ремонт в туалете, сменить пол в коридоре и обшить балкон. Но это уже не так интересно, думаю все с этим сталкивались. ;о)


воскресенье, 19 января 2014 г.

Arduino c dx.com

Захотел Гриша себе какой-нибудь интересный подарок на День Рождения. Недолго думая я полез в интернет в поисках чего-нибудь эдакого. Эдакое нашлось довольно быстро и называлось Arduino. Arduino - это открытая модульная система, с программируемым контроллером и возможностью собирать из нее практически все, что угодно - от датчика температуры или личной метеостанции публикующей показания датчиков в сети, до систем "умный дом". Ах да. Чуть не забыл. Волшебное слово "роботы" в списке возможных результатов тоже присутствует!

Перебрал довольно много интернет-магазинов, торгующих комплектующими, и остановился на Dealextreme

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

Все удовольствие стоило около 97 баксов. Кажется - дорого? Но этого робота можно модифицировать, навешивая на него новые датчики и платы расширения. Можно полностью переписать управляющий им код и получить... практически, что угодно.

Хорошая подборка литературы имеется на сайте RoboCraft, еще очень толковые форумы и подборки статей можно найти на сайтах Arduino.ru и Cxem.net

Ну и для тех, кто задал себе вопрос: "И что с этим делать?", - небольшой каталог датчиков с описанием функций

А себе я заказал контроллер и с десяток датчиков для экспериментов. И такую вот симпатичную кружку

Оплатил заказ через PayPal 16-го января. 19 января посылка была отдана на подготовку к отправке. Надеюсь до китайского НГ ее успеют отправить (НГ у Китайцев начинается 25 января и длится до 5 февраля. В эти дни большинство китайских интернет-магазинов не работают.)