Веб камера ардуино

Скетч для Arduino

#include <Adafruit_VC0706.h> #include <SPI.h> #include <SdFat.h> #include <SdFatUtil.h> #include <Ethernet.h> #include <SoftwareSerial.h> // Arduino Ethernet shield: pin 4 #define chipSelect 4 // Using SoftwareSerial (Arduino 1.0+) or NewSoftSerial (Arduino 0023 & prior): #if ARDUINO >= 100 // On Uno: camera TX connected to pin 2, camera RX to pin 3: SoftwareSerial cameraconnection = SoftwareSerial(2, 3); // On Mega: camera TX connected to pin 69 (A15), camera RX to pin 3: //SoftwareSerial cameraconnection = SoftwareSerial(69, 3); #else NewSoftSerial cameraconnection = NewSoftSerial(2, 3); #endif Adafruit_VC0706 cam = Adafruit_VC0706(&cameraconnection); /************ ETHERNET STUFF ************/ byte mac = {0x90,0xa2,0xda,0x0e,0xf7,0xfc}; byte ip = {192, 168, 0, 30}; char rootFileName = «index.htm»; EthernetServer server(80); /************ SDCARD STUFF ************/ SdFat SD; Sd2Card card; SdVolume volume; SdFile root; SdFile file; // store error strings in flash to save RAM #define error(s) error_P(PSTR(s)) void error_P(const char* str) { PgmPrint(«error: «); SerialPrintln_P(str); if (card.errorCode()) { PgmPrint(«SD error: «); Serial.print(card.errorCode(), HEX); Serial.print(‘,’); Serial.println(card.errorData(), HEX); } while(1); } /**********************SETUP()*********************/ void setup() { #if !defined(SOFTWARE_SPI) #if defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__) if(chipSelect != 53) pinMode(53, OUTPUT); // SS on Mega #else if(chipSelect != 10) pinMode(10, OUTPUT); // SS on Uno, etc. #endif #endif PgmPrint(«Free RAM: «); Serial.println(FreeRam()); pinMode(10, OUTPUT); digitalWrite(10, HIGH); if (!card.init(SPI_HALF_SPEED, 4)) error(«card.init failed!»); if (!volume.init(&card)) error(«vol.init failed!»); PgmPrint(«Volume is FAT»); Serial.println(volume.fatType(),DEC); Serial.println(); if (!root.openRoot(&volume)) error(«openRoot failed»); PgmPrintln(«Files found in root:»); root.ls(LS_DATE | LS_SIZE); Serial.println(); PgmPrintln(«Files found in all dirs:»); root.ls(LS_R); Serial.println(); PgmPrintln(«Done»); Serial.begin(9600); Serial.println(«VC0706 Camera snapshot test»); if (!SD.begin(chipSelect)) { Serial.println(«Card failed, or not present»); return; } Ethernet.begin(mac, ip); server.begin(); } #define BUFSIZ 100 /**********************LOOP()*********************/ void loop() { char clientline; char *filename; int index = 0; int image = 0; EthernetClient client = server.available(); if (client) { boolean current_line_is_blank = true; index = 0; while (client.connected()) { if (client.available()) { char c = client.read(); if (c != ‘\n’ && c != ‘\r’) { clientline = c; index++; if (index >= BUFSIZ) index = BUFSIZ -1; continue; } clientline = 0; filename = 0; Serial.println(clientline); if (strstr(clientline, «GET / «) != 0) { filename = rootFileName; } if (strstr(clientline, «GET /») != 0) { if (!filename) filename = clientline + 5; (strstr(clientline, » HTTP»)) = 0; Serial.println(filename); if (strstr(filename, «tttt») != 0) takingPicture(); if (strstr(filename, «rrrr») != 0) removePicture(); if (! file.open(filename, O_READ)) { client.println(«HTTP/1.1 404 Not Found»); client.println(«Content-Type: text/html»); client.println(); client.println(«<h2>File Not Found!</h2>»); break; } Serial.println(«Open!»); client.println(«HTTP/1.1 200 OK»); if(strstr(filename, «.css») != 0) client.println(«Content-Type: text/css»); else if (strstr(filename, «.htm») != 0) client.println(«Content-Type: text/html»); else if (strstr(filename, «.jpg») != 0) client.println(«Content-Type: image/jpeg»); client.println(); int16_t c; while ((c = file.read()) >= 0) { Serial.print((char)c); client.print((char)c); } file.close(); }else { // everything else is a 404 client.println(«HTTP/1.1 404 Not Found»); client.println(«Content-Type: text/html»); client.println(); client.println(«<h2>File Not Found!</h2>»); } break; } } delay(1000); client.stop(); } } /**********************FUNCTION1()*********************/ void takingPicture(){ if (cam.begin()) { Serial.println(«Camera Found:»); } else { Serial.println(«No camera found?»); return; } char *reply = cam.getVersion(); if (reply == 0) { Serial.print(«Failed to get version»); } else { Serial.println(«——————«); Serial.print(reply); Serial.println(«——————«); } //cam.setImageSize(VC0706_640x480); // biggest cam.setImageSize(VC0706_320x240); // medium //cam.setImageSize(VC0706_160x120); // small uint8_t imgsize = cam.getImageSize(); Serial.print(«Image size: «); if (imgsize == VC0706_640x480) Serial.println(«640×480»); if (imgsize == VC0706_320x240) Serial.println(«320×240»); if (imgsize == VC0706_160x120) Serial.println(«160×120»); Serial.println(«Snap in 3 secs…»); delay(3000); cam.takePicture(); Serial.println(«taking picture»); file.open(«IMAGE.jpg», O_RDWR | O_CREAT); uint16_t jpglen = cam.frameLength(); pinMode(8, OUTPUT); byte wCount = 0; while (jpglen > 0) { uint8_t *buffer; uint8_t bytesToRead = min(32, jpglen); buffer = cam.readPicture(bytesToRead); file.write(buffer, bytesToRead); if(++wCount >= 64) { Serial.print(‘.’); wCount = 0; } jpglen -= bytesToRead; } file.close(); Serial.println(«end of taking picture»); } /**********************FUNCTION2()*********************/ void removePicture(){ file.open(«IMAGE.jpg», O_READ | O_WRITE); file.remove(); }

Некоторые необходимые комментарии к коду приведены ниже.

1. Надо изменить mac и ip в соответсвии с информацией о вашем Arduino.

2. Для того, чтобы отобразить jpg файл в браузере, его надо преобразовать в jpg формат, для этого надо прописать следующие строки

client.println(«HTTP/1.1 200OK»);

client.println(«Content-Type: image/jpeg»);

client.println();

client.print((char)c);

>Запуск и проверка работоспособности

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

2. Для того, чтобы отобразить снимок, укажите в строке браузера:

Сверхдешёвая камера с управлением — своими руками


В качестве преамбулы скажу, что поскольку, подобная статья уже была, я по-началу и не собирался писать о камере на хабр.
Камера была сделана, просто ради интереса, тренировки навыков и отработки технологии удалённого управления физическими объектами.
Немного позже использовал эту систему как наглядный пример, сопровождающий обзор, на другом ресурсе (а фактически, в качестве чита что-бы подтянуть голоса). Это был настоящий бета-тест, который выявил кучу недоработок. За это огромное спасибо всем неравнодушным, помогавшим кто советом, а кто и куском кода.
При этом, посетители, которые игрались с камерой, помимо того, что не проходило и часу чтобы не помянули хабр, так ещё и начали активно интересоваться устройством всего этого дела, техническими деталями, программной реализацией и конечно же стоимостью.
Вот это всё и сподвигло меня на статью. А чтобы не повторюшничать, я и решил заостриться на стоимости, т.к. у автора предыдущего топика на эту тему, насколько я помню, итоговая стоимость вылилась во что-то в районе 5000р.
О том какова стоимость моей поделки: читаем ниже.
Итак, «как корабль назовешь, так он и поплывет», раз написал заголовок про дешевизну — буду соответствовать прозой, так что, детали — потом, а сперва о стоимости.

Пройдём по ценам*

* все цены даны со скидками. О скидках — отдельно, пожже.
Необходимый набор:

  • Arduino Uno (или nano) — 15.29$
  • Сервопривод (рулевая машинка) SG90 2х2.37$ = 4.74$
  • Вебкамера 3.01$

Итого: 23.04$ (примерно 750р, на данный момент)
Дополнительный набор (ленивости + плюшки):

  • Экран от Nokia 5110 — 250р (в комплекте с самой Нокией и блоком питания, куплено пару лет назад, будем считать, что именно для этой цели, реально можно найти уже вдвое дешевле, или вообще на халяву)
  • MegaShield v4 к Arduino — 5.86$
  • Проводки-коннекторы — 2.86$ (40 штук за эту цену, реально использовано 7)
  • Сверхяркий сверхсиний сверхтодиод для подсветки экрана — 5р/шт (лучше 4шт., у меня сделано неправильно)

Итого: примерно 550р
Всего: 1300р

О реализации

Всё делалось с нуля. Повторять то, что уже было — я не стал, во-первых из соображений тренировки, а во-вторых Ethernet-модуля у меня на тот момент не было, я решил что это всё слишком сложно (там был завязан MySQL) и это решение мне однозначно не подойдёт.
О задачах
Задачи я себе обозначил следующие:

  • Видеть картинку/видео
  • Иметь возможность управлять камерой
  • Иметь возможность управлять размером и качеством видео или картинки, причём не «уже на стороне клиента», а «ещё на стороне сервера, по команде клиента». Такая необходимость возникла из-за того что мне не везде доступен «большой и широкий интырнет»
  • Обеспечивать приемлемую «реалтаймовость»
  • Иметь задел на будущее — управление нагрузкой 220В и т.д. Собственно ради этого всё и затевалось, т.к. готовые решения либо жутко дороги, либо такой возможности не предоставляют.

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

  • Видео либо грузит процессор в случает показа на несколько пользователей, либо даёт задержку 5-10, т.е. не обеспечивает «реалтаймовость», из-за чего нельзя сразу определить адекватность и вообще работоспособность управления
  • Использование сервиса трансляций, хотя и сильно разгружает сервер в случае большого онлайна, не обеспечивает необходимую надёжность, и, опять же, даёт задержку
  • Специальный сервер для трансляции видеопотока требует определённых навыков, которых у меня пока что нет
  • Использование отображения путём смены картинок не обеспечивает высокий fps, а также постоянно обращается к жёсткому диску, что, при большом количестве пользователей, может вызывать лаги картинки не из-за загрузки процессора, а именно из-за обращения к диску

О решениях

  • Решено использовать в качестве отображения — картинки
  • Для исключения жёсткого диска из процесса выдачи картинок установлен RamDisk, на который дважды в секунду «ложится» изображение с вебкамеры
  • Для выдачи картинки решено использовать php и gdlib
  • Обновление картинки инициируется клиентом посредством javascript и ajax, и происходит без обновления самой странички

Довольно лирики!

Как выглядит

Выглядит всё более чем скромно
Работает примерно так:

Ардуина, если кто не видел

Мегашилд с проводками

«Сэндвич» в профиль

«Сэндвич» анфас

LCDшка

Она же вид сзади (пины и кондёр)

В сборе

В сборе 2
Колхоз — система проводков и верёвочек (крепление камеры)
Куда же без него

Как устроено аппаратно

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

Как работает программно

На стороне клиента чистый веб-интерфейс, без всяких плагинов и примочек. Только html, css, и javascript (+ajax).
На стороне сервера

  • Сам сервер — Apache
  • Обработчик скриптов — php
  • Приём картинок с камеры — любая самая простая доступная, бесплатная или самопальная программа для сохранения картинок с вебкамеры
  • Хранение картинки — RamDisk, утилита для создания дискового раздела в оперативной памяти (русскоязычная версия RAMDisk «Enterprise» бесплатна для локализованных систем)
  • Чтобы не прописывать в php прямых локальных путей, папка с картинкой смонтирована в www папку с помощью juction (бесплатная утилита Марка Руссиновича)
  • Передача управления из интернета к Arduino реализована с помощью программы-прокси, следующим образом: php скрипт создаёт UDP сокет и отправляет датаграмму на определённый порт, далее программа-прокси слушает этот этот порт и принимает приходящие на него сообщения и отправляет их на COM-порт Arduino (можно даже без обработки). Выбор UDP вызван исключительно для упрощения системы, UDP не требует никаких подтверждений и проверок о доставке-отправке ни со стороны клиента, ни со стороны сервера.

На стороне Arduino

  • Сама (почему «сама»? потому что «плата») Arduino
  • Скетч внутри неё — стандартные примеры из штатного набора arduino-0022 servo и serial + найденная на просторах интернета библиотека для дисплея, доработанная до приемлемого вида (в плане кириллицы и латиницы одновременно)
  • На данный момент плюсом стоит мегашилд, чисто из-за удобства и культурного вида — в этом варианте я не спаял ни одного проводка (за исключением платы к дисплею)

Система выдержала все нашествия и рейды, а так же онлайн более 120 пользователей.
Были случаи отказа управления, которые случались из-за моих недоработок в программе-прокси, в частности из-за недостаточной обработки ошибок, в то время как програмная часть со стороны Апача и Ардуино держалась достойно.

Будьте внимательны к мелочам

Хочу отметить проблемы с программной частью которые случались из-за собственной невнимательности/неосведомлённости/ненаблюдательности:

  • Первое с чем я серьёзно мучался: Arduino принимает из отправленной на её виртуальный COM-порт строки отдельно первый байт и отдельно всё остальное. Какие изощрения я только не пробовал — и с массивами и с кучей проверок… Хоть ты убейся. Решение проблемы? Пришло неожиданно и внезапно, в моментк огда я об этом и не думал: Sleep 2 после чтения каждого байта. ВСЁ!
  • Вторая проблема — серьёзная нагрузка на сервер, казалось бы, из ничего, возникла потому, что обновление картинки было сделано по таймеру, не дожидаясь собственно факта загрузки картинки (или ошибки загрузки). Таком образом отсылалась куча «лишних» запросов.
  • Третье: FireFox оказался самым правильным и капризным браузером, и заставил меня учиться писать валидный код. Так например, событие OnClick по элементу Option работать не должно. А оно работет, везде кроме огнелиса.
  • Четвёртая, совершенно не явная и редко всплывающая: периодически картинка «ломалась». Как выяснилось, это происходило в момент когда файл был занят при записи. Т.е. проверка file_exists() проходила, а файл оставался залоченным. Не помогла и проверка is_writable(). Пришлось организовывать цикл по наличию ошибки и внутри него отрабатывать чтение файла «до победного конца».

Оставшиеся недоработки

Есть и такие.

  • Во-первых, это описанные в каментах «левые» символы иногда появляющиеся в конце сообщения на экране. На самом деле это команды управления. Уши этого бага растут из того что если активно спамить или жать кнопки, буфер ком-порта не успевает полностью прочитаться Ардуиной и последующие сообщения валятся в конец буфера. Решение есть, но пока не сделано.
  • Во-вторых, это периодическое падение UDP-сокета в программе прокси при большом онлайне. В чём причина — не знаю. Проявляется не сразу. Умирает и не «откисает». Помогает закрытие сокета и бинд по-новой. Возможно, виноват кривой видовский winsock.ocx. Переписывать это дело на API в бейсике, как-то лень. Пока одним из «топорных» решений вижу сброс и ребинд сокета по таймеру, каждые, скажем, полчаса.

О скидках

Общеизвестно, что в Китае — дешевле. Главное знать места, где именно дешевле, и как добыть дополнительную скидку.
Тут смысла писать нет — слишком большой объём текста с картинками, к тому же известный большинству.
Поэтому дабы не провоцировать ярых противников борьбы со спамом и прочим «реферальством», отмечу необходимый минимум — это скидка 15% на BiC, складывающаяся из одноразового купона на 10% и скидки за первую покупку 5% при вводе рекомендателя + хинт, как использовать эту систему неоднократно.
Все заинтересовавшиеся, могут ознакомиться с полной информацией по ссылке на страничке с, собственно, самой камерой.

Ссылки

Камера, работающий экземпляр, для тех кто ещё не видел и не наигрался
Топик, в котором ссылку на камеру слили на хабр раньше времени

Добавить комментарий

Ваш e-mail не будет опубликован. Обязательные поля помечены *