이번엔 기상측정기구와 이더넷 쉴드가 장착된 아두이노(Arduino)로 기상관측 자료를 트위터(Twitter)에 자동으로 올려주는 기특한 녀석을 소개시켜 드리겠습니다. 제법 시스템이라 불릴 만한 구성입니다.  바람의 방향과 속도 및 강수량을 측정할 수 있는 기상관측 기구로 부터 측정된 값을 모니터링하고 이를 곧바로 인터넷을 통해 트위터(twitter)에 올리는 것입니다.  기상 데이타 측정용으로 사용된

가상 관측기기에 대해선 관련글 링크

를 참고하시기 바랍니다.

본 글을 이해하기위해 필요한 사전지식:  기본적인 아두이노 스케치 개발환경 사용법, 라이브러리 설치법, 이더넷쉴드 사용경험, 트위터의 기본적인 이해, 기상관측 기구 센싱방법 등


트위터?
트위터(Twitter)가 뭔지는 저보다 여러분이 더 잘 알고 계실것 같습니다.  제경우 트위터 사용법도 잘 모릅니다만 순서가 뒤바껴서 트위터 포스팅해주는 로봇을 먼저 만들게 되었습니다;;;   이글을 이해하시려면 일단 트위터에 가입하셔서 트위(Tweet) 하나를 올려보시면 됩니다.

아두이노 + 이더넷 쉴드
썰렁한 아두이노(Arduino) 보드에 이더넷 쉴드를 얹으면 인터넷을 통해 전세계와 의사소통이 가능합니다. 더불어 천재적이면서 친절하기까지 하신 분들이 아두이노로 트위터에 메시지를 보내는 예제와 라이브러리를 공개해 주셨으므로 저와 여러분들은 손쉽게 엄청난?;; 시스템 구현을 할 수 있는 상황입니다.  이더넷 쉴드 사용 경험이 없으신 분들은 우선 공개된 정보와 아두이노 스케치IDE에 포한된 기본 예제를 참고하시기 바랍니다.


아두이노 + 이더넷 쉴드의 한계
이더넷 쉴드를 장착하면 아두이노 보드로 정보서비스를 제공해주는 서버 역활을 할 수 있습니다.  외부 접속이 가능한 IP주소를 사용한다면 전세계의 인터넷 접속가능 기기를 통해 정보를 제공받을 수 있습니다.  하지만 동시 접속수 제한이나 퍼포먼스 문제로 원할한 서비스 제공에는 한계가 있습니다. 이때문에 아두이노를 데이타 센싱이 용이한 현장에 설치 후 취합한 데이타를 웹서비스를 담당하는 PC급 서버로 전달하여 제대로된 웹서비스 환경에 응용하는 것이 더 안정적인 방법일 수 있습니다.

이때 손쉽게 사용가능한 서비스가 HTTP Client입니다. 즉, 아두이노는 웹클라이언트로써 웹서버에 필요할 때만 접속하여 데이타를 송/수신 하는 방식입니다.    이더넷 라이브러리와 함께 제공되는 기본예제(WebClient)를 조금만 수정하면 활용이 가능합니다.  IP주소로 접속이 가능한 웹서버가 있는경우 추천할만한 방법입니다.  다만 문제는 기본 이더넷 쉴드 라이브러리를 이용한 Web Client 로는 domain name ( http://robobob.co.kr 같은) 을 통한 서버 접속이 안되고  123.123.123.123  같은 숫자형식의 ip를 이용한 접속만 가능하다는 점입니다.  예전에는 ip주로를 통한 접속 주소를 지원하는 웹호스팅 서비스가 있었지만 요즘엔 이를 지원하지 않는곳이 대부분인것 같습니다. 때문에 특정 웹호스팅 서비스나 단독 호스팅서비스를 사용하지 않으면 Web Client 사용한 접속이 불가한 상황입니다.  (일반적인 URL 주소를 통한 아두이노의 HTTP Client 접속 방법을 아시는 분은 안내 부탁드립니다.)

트위터 활용
HTTP Client 활용에 문제가 있으나 오히려 더 효과적일 수 있는 데이타 공유 방법이 있으니 바로 트위터입니다. 짧은 단문을 포스팅하면 수많은 팔로워에게 전달되는 효과적인 데이타 공유시스템을 활용하는 것입니다. 게다가 요즘 대세인 스마트폰에서도 손쉽게 접근이 가능한 최신병기입니다. 그런데 방법이 어렵냐고요???  제가 해보니 생각보다 매우 간단했습니다.   트위터 엡 개발을 위해 API니 뭐니 습득하고 복잡한 절차가 필요한지 알았는데요, 그게 아니고 그냥 사용만 하면 되네요...


트위터 포스팅 방법
방법은 요 사이트에 잘 요약되어있습니다.

http://arduino-tweet.appspot.com/

별 설명이 필요없습니다만, 정리하자면 아래와 같습니다.
1. 트위터 앱(O Auth)에 접속하여 승인을 해주면 복잡한 문자열로 된 토큰 정보가 생성됩니다. 이 값을 복사하여 잘 보관합니다.
2. 관련 라이브러리들을 설치해줍니다. 두군데 정도 접속하여 파일을 받아서 아두이노 library 폴더에 적절한 이름으로 복사합니다.
3. 예제소스를 실행해봅니다.
   위 라이브러리를 제대로 설치하면 스케치 IDE  Examples 리스트에 Twitter가 생깁니다.
   File menu > Examples > Twitter > SimplePost 예제를 선택해줍니다.

이더넷 쉴드 사용시 수정해주는  ip주소와 네트웍정보 몇개만 수정해주시고 추가로  "YOUR-TOKEN-HERE" 부분을  위에서 발급받은 토큰값으로 대체해 주면 준비 끝~ 입니다.
이젠 잘 전송되는지 확인을 위해 시리얼 모니터링 창을 열고 속도(9600)를 맞쳐줍니다.

SimplePost 예제 소스 내용보기

#if defined(ARDUINO) && ARDUINO > 18   // Arduino 0019 or later
#include <SPI.h>
#endif
#include <Ethernet.h>
#include <EthernetDNS.h>
#include <Twitter.h>

// Ethernet Shield Settings
// 쉴드 밑면에 있는 mac 정보를 적어줍니다. 쉴드가 하나인경우 안바꿔도 무방
byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED }; 


// substitute an address on your own network here
byte ip[] = { 192, 168, 2, 250 };   //공유기에서 사용되지않는 IP값을 지정

// Your Token to Tweet (get it from http://arduino-tweet.appspot.com/)
Twitter twitter("YOUR-TOKEN-HERE");   // 발급받은 토큰값을 이곳에 붙여넣으세요

// Message to post
char msg[] = "Hello, World! I'm Arduino!";   // 한글 전송도 가능합니다!

void setup()
{
  delay(1000);
  Ethernet.begin(mac, ip);
  Serial.begin(9600);
 
  Serial.println("connecting ...");
//부팅시 1회 포스팅 시도합니다.
  if (twitter.post(msg)) {
    // Specify &Serial to output received response to Serial.
    // If no output is required, you can just omit the argument, e.g.
    // int status = twitter.wait();
    int status = twitter.wait(&Serial);
    if (status == 200) {
      Serial.println("OK.");
    } else {
      Serial.print("failed : code ");
      Serial.println(status);
    }
  } else {
    Serial.println("connection failed.");
  }
}

void loop()
{ //아무것도 안합니다. 
}

SimplePost 예제 소개
예제로 사용된 소스에는 아두이노가 부팅할때마다 1회 포스팅을 하고, 포스팅 될때마다 결과 상태를 시리얼 통신으로 전달해 줍니다. (참고로 시리얼 창 새로 열거나 통신속도를 변경하여도 아두이노가 재부팅이되어 새로 포스팅을 시도합니다.)
결과가 Okay로 뜬다면 잠시 후 여러분의 트위터 페이지에 새 트윗이 올라온것이 보일 겁니다.
Okay 외의 오류 메시지를 잘 살펴보시면 디버깅에 도움이 됩니다.

포스팅 시도후 오류메시지가 뜨는 경우가 종종 있는데요  제가 발견한 주요 오류 내용은 아래와 같습니다.
1. 동일한 내용을 중복하여 전송한 경우: 이경우 중복 전송된 값은 무시됩니다.
2. 트윗 작성 한계치 초과 :  시스템 보호를 위해 일정 시간당 한정된 개수의 트윗만 허용됩니다.   이 수량을 초과시 일정기간 동안 트위 작성이 불가하며 일정 시간이 지난뒤에 다시 작성이 가능해집니다.   가령  수십초에 한번씩 계속 트윗을 하신다면 얼마지나지 않아 트윗 등록이 거부될 겁니다.   아래의 예제에서는 1시간에 1회 트윗을 발신하게됩니다.




드디어 트위터 기상로봇

이제 포스팅될 정보를 여러분의 용도에 맞게 잘 구성해주시면 여러분만의 활용도 만빵인 트위터 포스팅 봇이 완성됩니다.
제 경우 이미 소개드렸었던 기상측정기기를 활용하여 1시간마다 기상데이타를 측정하여 전송해주는 예제를 만들어봤습니다.
실제 포스팅된 내용들을 아래의 페이지에서 확인 가능합니다.

로보밥 기상측정 로봇 트위터 메시지 보러가기 :  

http://twitter.com/artrobot_bot1

한동안 실내 테스트만 하다가 2012년 1월 29일 현재,  옥외 설치하여 운영중입니다.  위 링크에 가보시면 실제 기상자료를 참고하실 수 있습니다. (경기도 수원시 오목천동)

위에서 트윗된 메시지의 포스팅 시간을 보시면 (오차가 1분도 안되네요^^.)1시간 간격으로 전송이 된것을 보실 수 있습니다. 하지만 가끔씩  몇 시간에 한번만 트윗된 것도 보이실 겁니다. 이때는 테스트를 위해 실내에서 측정된 풍량/풍속/강수량 수치의 변화가 없어서 동일한 내용을 반복 포스팅한것으로 인지되어 무시된 경우입니다.  (계속 변화하는 시간값이나 랜덤숫자를 함께 전송하시면 이문제가 해결됩니다.)

사진. 기상측정 기구

+

사진. 아두이노로 만든 트위터 포스팅해주는 로봇

사용된 예제 소스 소개
값의 정밀도나 완성도를 무시하고 필요하신 분들에게 참고가 될 수 있도록 예제소스를 공개합니다.  풍향, 풍속, 강수량의 측정치 산정의 기준은 기상측정기기 제조사에서 제공한 데이타 시트를 참고하여 계산된 값입니다. 센서 연결 방법 및 기본 사용법 소개는 관련글 링크에 있는 글을 참고하시기 바랍니다.


/*
* 풍향, 풍속, 강수량을 Twitter에 포스팅하는 예제
  수정일시:  (2011.09.08a)  디버깅 및 기능개선을 위해 수시로 변경될 수 있습니다.

 풍향: 포스팅시 1회 계산 (16방위 중 하나로 측정)
 풍속: 20ms 주기로 10초간 모니터링 후 풍속계산
 강수량: 20ms 주기로 모니터링하여 1시간동안 누적(1시간에 1회 공식 데이타로 사용)
 
 핀연결
 풍향계: A0   풍향센서 전선 2개 중 하나는 GND에 나머지선은 A0에 연결 및 10k저항 거쳐 5V에 연결
 풍속계: D2   풍속센서 전선 2개 중 하나는 D2에 나머지는 GND
 강수계: D3   강수센서 전선 2개 중 하나는 D3에 나머지는 GND

 기타 핀 연결 (없어도 무방)
 D7: 풍속계 상태 LED
 D8: 강수량계 상태 LED
 D9: 버튼 ( 버튼을 누르면 10초내로 즉시 포스팅)

* http://ArtRobot.co.kr
* http://RoboBob.co.kr
*/

#include <SPI.h>
#include <Ethernet.h>
#include <EthernetDNS.h>
#include <Twitter.h>
#include <stdio.h>

#define WIND_N    0
#define WIND_NNE  22.5
#define WIND_NE    45
#define WIND_ENE  67.5
#define WIND_E    90
#define WIND_ESE  112.5
#define WIND_SE    135
#define WIND_SSE  157.5
#define WIND_S    180
#define WIND_SSW  202.5
#define WIND_SW   225
#define WIND_WSW  247.5
#define WIND_W    270
#define WIND_WNW  292.5
#define WIND_NW   315
#define WIND_NNW  337.5

// Ethernet Shield Settings
byte mac[] = { 0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xED }; //바꿔주세요

// substitute an address on your own network here
byte ip[] = { 192, 168, 100, 55 }; //바꿔주세요

// Your Token to Tweet (get it from http://arduino-tweet.appspot.com/)
Twitter twitter("발급받으신 토큰값을 적어주세요");  //바꿔주세요

// Message to post
char msg[100] = "";

const int windVanePin = A0;  // 풍량계 센서 입력핀
int windVaneValue = 0;        //
float windSpeed = 0;
float rainGauge = 0;
float windDirection = 0;
char windName[4];  // N (North)  S South,   NS(north south)   NNS(north north south) etc

  unsigned long windSpeedTimer;
  int windSpeedState = true;
  int windSpeedPin = 2;
  int windSpeedCounter = 0;
  unsigned long rainGaugeTimer;
  int rainGaugeState = true;
  int rainGaugePin = 3;
  int rainGaugeCounter = 0;
  int buttonState = true;
  int buttonPin = 9;
  int postTrigger = false;
  int postingCounter = 0;

void setup()
{
  delay(1000);
  Ethernet.begin(mac, ip);
  Serial.begin(9600);
 
  //wind speed sensor
  pinMode(windSpeedPin, INPUT);
  digitalWrite( windSpeedPin, HIGH);
 
  //rain gauge sensor
  pinMode(rainGaugePin, INPUT);
  digitalWrite( rainGaugePin, HIGH);

  //button for postMessge
  pinMode(buttonPin, INPUT);
  digitalWrite(buttonPin, HIGH);
 
  //wind speed sensor status LED
  pinMode(7, OUTPUT);
  digitalWrite(8,HIGH);
 
  //rain gauge status LED
  pinMode(8, OUTPUT);
  digitalWrite(8,HIGH);
 
}

// 문자배열변수값을 시리얼로 확인할경우만 사용
void printChars(char *msg, int len){
  if(len == 0) return;
  for(int i=0; i<len ; i++)
    Serial.print( msg[i]);
  Serial.println();
}

// sprintf함수에서 float형 print문제 대체용 함수 , float의 정수부인쇄
int f2h( float num){
  return int(num);
}

// sprintf함수에서 float형 print문제 대체용 함수 , float의 소수부인쇄
int f2p( float num){
  return (num - int(num)) * 100;
}


void loop()
{
  // 10초마다 샘플링 반복
  windSpeedTimer = millis() + 10000;
  windSpeedCounter = 0;
  while( millis() < windSpeedTimer ){
    delay(20); // 20ms 주기로 버튼이나 센서 스위칭 감시
    //wind speed
    if( (windSpeedState == true ) && !digitalRead( windSpeedPin) ){
        windSpeedCounter++;
        windSpeedState = false;
    }else if( (windSpeedState == false) && digitalRead(windSpeedPin) ){
        windSpeedState = true;
    }   
    //rain gauge
    if( (rainGaugeState == true ) && !digitalRead( rainGaugePin) ){
        rainGaugeCounter++;
        rainGaugeState = false;
    }else if( (rainGaugeState == false) && digitalRead(rainGaugePin) ){
        rainGaugeState = true;
    }       
   
    //button
    if( (buttonState == true ) && !digitalRead( buttonPin) ){
        postTrigger = true;
        buttonState = false;
    }else if( (buttonState == false) && digitalRead(buttonPin) ){
        buttonState = true;
    }      
    digitalWrite(7, windSpeedState);  
    digitalWrite(8, rainGaugeState);      
  }

  windSpeed = windSpeedCounter * 0.24;   // 1초당 1회 스위칭시 2.4km/h
  //Serial.print( "winSpeed: ");
  //Serial.println( windSpeed);

  rainGaugeTimer++;
  if( rainGaugeTimer > 360){  //1시간( 10 * 360 sec) 마다 postTrigger On
    rainGauge = rainGaugeCounter * 0.2794;
    rainGaugeCounter = 0;

    rainGaugeTimer = 0;
    postTrigger = true;
  }
  getWindDirection(); 

// 1시간마다 or 버튼이 눌릴때마다 Trigger 가 true가 되고 이때 포스팅된다.
  if( postTrigger ){ //post Message
    postTrigger = false;
    sprintf( msg, "[%d]기상측정로봇: 풍향:%s/%d.%d 풍속:%d.%d(km/h) 1시간당 강수량 %d.%d(mm/h).",postingCounter++ , windName, f2h(windDirection), f2p(windDirection), f2h(windSpeed), f2p(windSpeed),  f2h(rainGauge), f2p(rainGauge) );

    printChars( msg, sizeof(msg) );  //시리얼창으로 확인용, 생략가능
    postMessage();
  }

}

void postMessage(){
  Serial.println("connecting ...");
  if (twitter.post(msg)) {
    // Specify &Serial to output received response to Serial.
    // If no output is required, you can just omit the argument, e.g.
    // int status = twitter.wait();
    int status = twitter.wait(&Serial);
    if (status == 200) {
      Serial.println("OK.");
    } else {
      Serial.print("failed : code ");
      Serial.println(status);
    }
  } else {
    Serial.println("connection failed.");
  }
}

float getWindDirection(void){
 // 풍향센서의 출력 전압을 아날로그 센서로 읽어옴
  int readValue = analogRead(windVanePin);           
  // ADC입력치를 0~5V 범위로 변환, 데이타시트 값과 대응이 쉽도록.
  windVaneValue = map(readValue, 0, 1023, 0, 500); 
 
  // 해당 전압치에 따라 16가지 방향 중 하나의 풍향을 결정
  if( windVaneValue < 35){
    // DIR5  112.5  0.32v (31 32)    0~35
    windDirection = WIND_ESE;
    strcpy(windName ,"ESE");       
  }else if( windVaneValue < 43 ){
    // DIR3  67.5  0.41v(40 41)    ~42   
    windDirection = WIND_ENE;
    strcpy(windName , "ENE");       
  }else if( windVaneValue < 50 ){
    // DIR4  90  0.45v(44 45)      ~50   
    windDirection = WIND_E;
    strcpy(windName , "E");       
  }else if( windVaneValue < 70 ){
    // DIR7  157.5  0.62v(60 62)   ~70
    windDirection = WIND_SSE;
    strcpy(windName , "SSE");       
  }else if( windVaneValue < 100 ){
    // DIR6  135  0.90v(89 90)      ~100
    windDirection = WIND_SE; 
    strcpy(windName , "SE");       
  }else if( windVaneValue < 130 ){
    // DIR9  202.5  1.19v(119 120)  ~130
    windDirection = WIND_SSW; 
    strcpy(windName , "SSW");       
  }else if( windVaneValue < 170 ){
    // DIR8  180  1.40v(140 141)   ~170
    windDirection = WIND_S; 
    strcpy(windName , "S");       
  }else if( windVaneValue < 210 ){
    // DIR1  22.5  1.98v(198 199)  ~210
    windDirection = WIND_NNE; 
    strcpy(windName , "NNE");           
  }else if( windVaneValue < 250 ){
    // DIR2  45  2.25v(226 227)    ~250
    windDirection = WIND_NE; 
    strcpy(windName , "NE");           
  }else if( windVaneValue < 300 ){
    // DIR11 247.5  2.93v(293 294)  ~300
    windDirection = WIND_WSW; 
    strcpy(windName , "WSW");       
  }else if( windVaneValue < 320 ){
    // DIR10 225  3.08v(308 310)   ~320
    windDirection = WIND_SW; 
    strcpy(windName , "SW");       
  }else if( windVaneValue < 360 ){
    // DIR15 337.5  3.43 (343 345)  ~360
    windDirection = WIND_NNW; 
    strcpy(windName , "NNW");       
  }else if( windVaneValue < 395 ){
    // DIR0  0  3.84v(384~385)    ~395
    windDirection = WIND_N; 
    strcpy(windName , "N");
  }else if( windVaneValue < 415 ){
    // DIR13 292.5  4.04v(405 406)  ~415
    windDirection = WIND_WNW; 
    strcpy(windName , "WNW");   
  }else if( windVaneValue < 450 ){
    // DIR14 315  4.34(433 434)  ~450
    windDirection = WIND_NW; 
    strcpy(windName , "NW");       
  }else if( windVaneValue < 490 ){
    // DIR12 270  4.62v(461 463)    ~490
    windDirection = WIND_W; 
    strcpy(windName , "W");       
  }else{
   //error  알수없는 값범위
  }

}





관련제품
아두이노 (Arduino)
표준 아두이노 이더넷 쉴드
이더넷 일체형 아두이노
기상측정용 기구

관련글 링크

  • 기상측정 기구 + 아두이노로 기상대 만들어요 (Arduino + Weather Sensor Assembly )
  • 트위터(Twitter)에 자동 포스팅하는 기상관측 로봇 (현재글)
  • 기상측정 트위터 로봇 - 실외 설치편
  • 이더넷쉴드 관련   링크1   링크2

추가사항 (2011.11.29)
아두이노와 이더넷 쉴드를 하나로 결합한 Ethernet Pro 보드로 기상측정 트위터 기능을 구현할 수도 있습니다. 특히 PoE 케이블을 이용하시면 전원과 이더넷케이블을 하나로 줄일 수 있어 더욱편리합니다. 가령 위의 시스템을 아래의 이미지와 같이 간소화 시킬 수 있습니다.

안녕하세요 재밌는 창작재료가 너무 많아 무엇부터 갖고 놀아야 할지 고민이 됩니다.  ^^.  즐거운 고민이죠;;
오늘은 기상측정용 기구를 소개해 드리겠습니다.

참고로,

다음글

에서는 기상측정기구와 이더넷 연동되는 아두이노(Arduino)로 기상관측 자료를 트위터(Twitter)에 자동으로 올려주는 기특한 녀석을 소개시켜 드리겠습니다. 제법 시스템이라 불릴 만한 구성입니다.  바람의 방향과 속도 및 강수량을 측정할 수 있는 기상관측 기구로 부터 측정된 값을 모니터링하고 이를 곧바로 이더넷 쉴드가 장착된 아두이노로 트위터(twitter)에 올리는 것입니다.

일단은 기상측정기구에 대해 좀더 알아봅시다! 

 

 

Weather Meter (기상 관측기구 , Weather Sensor Assembly)
본 장비는 3가지 기상 관측 기초 자료를 측정할 수 있습니다.


1. Rain Gauge (강수량계)
입수되는 수량에 비례하여 스위칭을 해주는 특수 기구를 이용하여 1회 스위치시마다 0.2794mm 의 강수량에 대응되게 됩니다. 물시계 작동원리를 응용한것인데 아이디어가 참 좋네요

사진에는 안나왔네요;;  일단, 동영상을 참고하시길~

2. Anemometer (풍속계)
회전시마다 일정수의 스위칭이 이뤄지며 기준 시간당 스위칭 수를 측정하여 풍속 계산이 가능합니다. 1초 동안 1회 스위칭시 2.4km/h ( 0.6666m/s)의 속력에 대응합니다.

 

 

 

 

 

3. Wind Vane ( 풍향계)
바람의 방향을 16방위로 측정 가능한 센서입니다. 내장된 저항들과 회전각에 따라 작동되는 스위치에 의해 내부저항 값이 변화하며, 이 값을 10k옴 저항 한개와 기준전압 5V를 통해 아두이노 아나로그 입력 핀으로 감지하게 됩니다.  회전각에 따른 출력 전압 값은 데이타시트에 나와 있습니다.  ( 회로연결법은 매우 간단합니다.  광센서나 온도센서의 저항 변화를 전압변화로 출력해주는 회로와 동일)

 

 

 

 

 

 

 

그림1. 풍속계의 내부 회로도( 저항과 스위치 들이 보입니다. 스위치가 눌려지면 내부저항이 변화함)

 

 

 

그림2. 풍향계 연결법( 5V와 10k저항을 사용하면 아래의 테이블에 있는 전압값이 Output핀에 출력됩니다.)

가령 0도(North,정북)을 향할 경우 내부저항은 약 33k옴이고  이때 10k옴 저항과 5V에 연결시 약 3.84V 가 측정됩니다.

Direction esistance Voltage
(Degrees) (Ohms) (V=5v, R=10k)
0 33k 3.84v
22.5 6.57k 1.98v
45 8.2k 2.25v
67.5 891 0.41v
90 1k 0.45v
112.5 688 0.32v
135 2.2k 0.90v
157.5 1.41k 0.62v
180 3.9k 1.40v
202.5 3.14k 1.19v
225 16k 3.08v
247.5 14.12k 2.93v
270 120k 4.62v
292.5 42.12k 4.04v
315 64.9k 4.78v(4.34v)
337.5 21.88k 3.43v

          
테이블1. 회전각/저항/전압 관계

참고. 제 경우 실측결과 315도 경우의 측정전압이 데이타시트와 달랐습니다. 참고하시기 바랍니다.


측정기의 비밀
본 장치를 회전시켜보면 풍향계와 풍속계 모두 매우 매끄럽게 회전됩니다. 센싱을 위해 스위치를 눌러주는 방식이라면 회전을 조금이라도 방해를 하게 될 테지만 이 기구는 그렇지 않습니다.  왜 그런지는 속을 들여다보면 나옵니다.  바로 마그네틱 센서입니다. 초소형 고강도 자석을 회전부에 설치해두고 회전시마다 자력으로 자석스위치를 일시적으로 단락시키는 방식을 사용하네요.  단순하고도 사용하기 편리한 메카니즘입니다.

연결 방법
풍향/풍속/강우량계 모두 각각의 케이블이 있으므로 총 3개의 케이블이 있습니다. 하지만 이중 풍속계 케이블을 풍향계 하단의 단자에 결합하게 되어 있으므로 2개의 케이블만 사용하면 됩니다.  케이블 전선은 일반 전화선과 유사하고 단자는 인터넷 케이블에 사용되는 것과 동일한 RJ45-8pin 단자입니다. 즉, 인터넷 케이블과 결합소켓을 이용하여 원하는 길이로 늘려줄 수 있게됩니다.  이더넷 케이블을 절단하여 라인을 브레드 보드에 결합하면 손쉽게 센서 스위치와 연결할 수 있습니다.(사진참조)
풍향계+풍속계 케이블 한곳에서 4개의 라인,  강수량계 케이블에서 2라인만 뽑아서 사용하면 됩니다.

 

 

측정 방법
풍속계와 강수량계는 스위치가 ON되는 횟수를 카운팅하면 되며 아두이노(Arduino)나 MCU를 이용하여 디지탈 입력핀으로 버튼이 눌려진 횟수를 카운팅하는 방법으로 측정이 가능합니다.  단, 스위칭시 노이즈로인해 1회 스위칭이 수~수십회 단락된 것으로 인지되므로 적절한 샘플링 주기(가령 20ms)마다 측정을 하여 스위치의 상태가 변이하는 순간에만 카운팅을 하도록 프로그램해줘야 합니다.  또한 강수량은 일정기간(가령 1시간)동안 그 값을 누적하여 결과로 사용하고,  풍속계도 일정 시간동안의 스위칭 횟수를 카운트하여 속력을 계산할 필요가 있습니다.
풍향 측정시엔 1개의 아날로그 입력핀이 필요하며, if else 구문으로 16 구간의 범위 조건으로 나눠서 각도를 구분해주면 됩니다. 


예제 소스
막상 어떻게 측정이 가능한지 궁굼하신 분들을 위해 참고용 소스코드를 함께 수록합니다.
정확성은 보장 못드리지만 참고하시기 바랍니다.

/*
 풍향, 풍속, 강수량 측정 예제
 
 풍향: 10초마다 1회 계산 (16방위 중 하나로 측정)
 풍속: 20ms 주기로 10초간 모니터링 후 풍속계산
 강수량: 20ms 주기로 모니터링하여 1시간동안 누적(1시간에 1회 공식 데이타로 사용)
 
 핀연결
 풍향계: A0   풍향센서 전선 2개 중 하나는 GND에 나머지선은 A0에 연결 및 10k저항 거쳐 5V에 연결
 풍속계: D2   풍속센서 전선 2개 중 하나는 D2에 나머지는 GND
 강수계: D3   강수센서 전선 2개 중 하나는 D3에 나머지는 GND
 
 http://ArtRobot.co.kr
 http://RoboBob.co.kr
*/


#define WIND_N    0 //정 북향
#define WIND_NNE  22.5
#define WIND_NE    45
#define WIND_ENE  67.5
#define WIND_E    90  //정 동향
#define WIND_ESE  112.5
#define WIND_SE    135
#define WIND_SSE  157.5
#define WIND_S    180
#define WIND_SSW  202.5
#define WIND_SW   225
#define WIND_WSW  247.5
#define WIND_W    270
#define WIND_WNW  292.5
#define WIND_NW   315
#define WIND_NNW  337.5


// Message to post
char msg[100] = "";

const int windVanePin = A0;  // 아날로그 0번핀에 연결
int winVaneValue = 0;        //
float windSpeed = 0;
float rainGauge = 0;
float windDirection = 0;
char windName[4];  // N (North)  S South,   NS(north south)   NNS(north north south)

  unsigned long windSpeedTimer;
  int windSpeedState = true;
  int windSpeedPin = 2;
  int windSpeedCounter = 0;
  unsigned long rainGaugeTimer;
  int rainGaugeState = true;
  int rainGaugePin = 3;
  int rainGaugeCounter = 0;


void setup()
{

  Serial.begin(9600);
 
  //wind speed sensor
  pinMode(windSpeedPin, INPUT);
  digitalWrite( windSpeedPin, HIGH);
 
  //rain gauge sensor
  pinMode(rainGaugePin, INPUT);
  digitalWrite( rainGaugePin, HIGH);
 
}

/*
* char 배열의 문자열정보를 Serial통해 문자로 전송 ,  PC에서 참고용
* char 배열을 사용한것은 차후 인터넷통신(트위터전송)을 위한 것입니다.
*/
void printChars(char *msg, int len){
  if(len == 0) return;
  for(int i=0; i<len ; i++)
    Serial.print( msg[i]);
  Serial.println();
}

/*
* f2h와 f2p는  float형의 정수부와 소수부를 위한 것입니다.
* sprintf에서 float형 사용에 문제가 있어서 땜빵으로 사용된 함수입니다.
*/
int f2h( float num){
  return int(num);
}

int f2p( float num){
  return (num - int(num)) * 100;
}


void loop()
{
  // 10초마다 측정결과를 PC에 전달

  windSpeedTimer = millis() + 10000; // 10초를 주기로 반복됩니다.
  windSpeedCounter = 0;
  while( millis() < windSpeedTimer ){
    delay(20); //20ms 주기로 센서의 스위칭을 감지합니다.
    //wind speed  풍속계 센서 감지부
    if( (windSpeedState == true ) && !digitalRead( windSpeedPin) ){
        windSpeedCounter++;     //스위치 상태가 high에서 low로 떨어지는 순간을 감지합니다.
        windSpeedState = false;
    }else if( (windSpeedState == false) && digitalRead(windSpeedPin) ){
        windSpeedState = true;
    }   
    //rain gauge
    if( (rainGaugeState == true ) && !digitalRead( rainGaugePin) ){
        rainGaugeCounter++;     //스위치 상태가 high에서 low로 떨어지는 순간을 감지합니다.
        rainGaugeState = false;
    }else if( (rainGaugeState == false) && digitalRead(rainGaugePin) ){
        rainGaugeState = true;
    }       
  }
 
  rainGaugeTimer++;
 
  if( rainGaugeTimer > 360){// 10초 * 360 = 3600초(1시간)  지난 1시간동안 누적한 강수계 카운터로 강수량 계산
    rainGauge = rainGaugeCounter * 0.2794;
    rainGaugeCounter = 0;
    rainGaugeTimer = 0;
  }
  
    windSpeed = windSpeedCounter * 0.24 ;   // 1초당 1회 스위칭시 2.4km/h 속력이며 10초 기간이므로 0.24가 됨
    getWindDirection();  // 풍향은 발표시점에 1회만 측정
   
    //char 배열에 정보를 취합(인터넷 전송에 적합한 자료형)
    sprintf( msg, "ArtRobot's Weather Bot said => Wind: %s/%d.%d, %d.%d(km/h) Rain %d.%d(mm/h).", windName, f2h(windDirection), f2p(windDirection), f2h(windSpeed), f2p(windSpeed),  f2h(rainGauge), f2p(rainGauge) );
    printChars( msg, sizeof(msg) ); //PC에서 참고용으로 시리얼전송
    //이후 소개할 예제에서 위 자료를 이더넷을 통해 트위터에 포스팅하게됨
 
}


/*
* 풍향측정: 아날로그핀입력되는 전압값으로 16가지 방향중 하나로 계산됨.
*
*/
float getWindDirection(void){
  int readValue = analogRead(windVanePin);           
  // 0~1023 사이의 입력값을 0~5V 기준 값으로 비례변경함.
  winVaneValue = map(readValue, 0, 1023, 0, 500);   
  //이제 winVaneValue는 데이타 시트정보상의 방위별 전압치와 유사한 값이 됩니다.
 
  if( winVaneValue < 35){
    //0~0.35V를 동동서 로 인식함
    //   112.5  0.32v (31 32)    0~35
    windDirection = WIND_ESE;
    strcpy(windName ,"ESE");       
  }else if( winVaneValue < 43 ){
    //   67.5  0.41v(40 41)    ~42   
    windDirection = WIND_ENE;
    strcpy(windName , "ENE");       
  }else if( winVaneValue < 50 ){
    //   90  0.45v(44 45)      ~50   
    windDirection = WIND_E;
    strcpy(windName , "E");       
  }else if( winVaneValue < 70 ){
    //   157.5  0.62v(60 62)   ~70
    windDirection = WIND_SSE;
    strcpy(windName , "SSE");       
  }else if( winVaneValue < 100 ){
    //   135  0.90v(89 90)      ~100
    windDirection = WIND_SE; 
    strcpy(windName , "SE");       
  }else if( winVaneValue < 130 ){
    //   202.5  1.19v(119 120)  ~130
    windDirection = WIND_SSW; 
    strcpy(windName , "SSW");       
  }else if( winVaneValue < 170 ){
    //   180  1.40v(140 141)   ~170
    windDirection = WIND_S; 
    strcpy(windName , "S");       
  }else if( winVaneValue < 210 ){
    //   22.5  1.98v(198 199)  ~210
    windDirection = WIND_NNE; 
    strcpy(windName , "NNE");           
  }else if( winVaneValue < 250 ){
    //   45  2.25v(226 227)    ~250
    windDirection = WIND_NE; 
    strcpy(windName , "NE");           
  }else if( winVaneValue < 300 ){
    //  247.5  2.93v(293 294)  ~300
    windDirection = WIND_WSW; 
    strcpy(windName , "WSW");       
  }else if( winVaneValue < 320 ){
    //  225  3.08v(308 310)   ~320
    windDirection = WIND_SW; 
    strcpy(windName , "SW");       
  }else if( winVaneValue < 360 ){
    //  337.5  3.43 (343 345)  ~360
    windDirection = WIND_NNW; 
    strcpy(windName , "NNW");       
  }else if( winVaneValue < 395 ){
    //   0  3.84v(384~385)    ~395
    windDirection = WIND_N; 
    strcpy(windName , "N");
  }else if( winVaneValue < 415 ){
    //  292.5  4.04v(405 406)  ~415
    windDirection = WIND_WNW; 
    strcpy(windName , "WNW");   
  }else if( winVaneValue < 450 ){
    //  315  4.34(433 434)  ~450
    windDirection = WIND_NW; 
    strcpy(windName , "NW");       
  }else if( winVaneValue < 490 ){
    //  270  4.62v(461 463)    ~490
    windDirection = WIND_W; 
    strcpy(windName , "W");       
  }else{
   //error  알수없는 값범위
  }

}


결과 출력예
위 예제의 경우 10초마다 아래와 같은 결과가 PC로 전송됩니다.
ArtRobot's Weather Bot said => Wind: N/0.0, 3.83(km/h) Rain 0.0(mm/h).
ArtRobot's Weather Bot said => Wind: W/270.0, 4.79(km/h) Rain 0.0(mm/h).
...



관련제품 링크
 

기상측정기구(Weather Meter)

 

아두이노 UNO(Arduino)

연관글 보러 바로가기
 

트위터(Twitter)에 기상측정 자료 올려주는 로봇 만들기

여러분이 획기적인 아이디어와 훌륭한 기술로 안드로이드용 앱을 개발하고 계시다면 아마도 최근 구글 IOIO행사에서 발표된 Open ADK (공개형 안드로이드용 엑세서리 개발킷)에 대해 들어보셨을 겁니다.

안드로이드용 오픈 엑세서리 개발 보드는 PC에 각종 USB 장치를 연결하여 확장된 기능을 사용할 수 있는것 처럼 안드로이드와 결합하여 사용가능한 하드웨어 장치를 개발하기위해 사용됩니다. 이미 많은 안드로이드 기기가 GPS, 가속센서, 자기 센서등을 내장한채 보급되고 있습니다만, Open ADK와 호환 개발 보드를 사용하면 그밖에 특화된 센서를 연결하거나 좀더 고급의 유저 인터페이스를 결합시킬 수 있게됩니다.  그렇다고 새로운 기기의 개발에만 필요한것이 아닙니다. 표준화된 규칙을 이용하여 기존에 개발되어 있는 제품들과도 손쉽게 연동이 가능해 지므로 큰 노력 없이도 안드로이드와 많은 엑세서리와 연동할 수 있게되어 하드웨어 개발자와 안드로이드 디바이스 및 앱 개발자 모두에게 효과적인 환경이 제공되는것 같습니다.

Open ADK 관련 정보는 공개되어 있으므로 관련 개발보드는 직접 개발하실 수도 있습니다. 하지만 많은 분들에겐 이미 상용화된 제품을 이용하시는것이 효율적일겁니다.  이번에 소개해 드릴 제품도 그중의 하나입니다.

 

 



안드로이드 요요  ( IOIO for Android )

해당제품에 있는 실크 이미지를 잘 보시면 안드로이드 로봇이 한손에 요요를 갖고 노는것이 보입니다.  IOIO를 yo-yo 라고 발음한다고 하는데 안드로이드와 함께 연결되어 IO(Input Output) 기능을 수행하는 녀석을 요요로 표현한것은 매우 적절한 비유인것 같습니다.  서두가 길었습니다.  이제 본격적으로 요요 보드를 소개해 드리겠습니다.

아래의 정보들은 http://codaset.com/ytai/ioio/wiki 에 있는 요요 Wiki 정보와 요요 제조사의 튜토리얼 정보를 참고로 하였습니다.   (변경안내:  위 codaset 링크를 비롯한 모든 자료가  https://github.com/ytai/ioio/wiki 로 이전되었으니 신규 사이트 정보를 이용해주시기 바랍니다.)

1부. 둘러보기

요요,  너의 정체가 무엇이냐?

제품에 떡 하니 자리를 차지하고있는 녀석은 MICROCHIP사의 PIC24FJ256 MCU입니다.   보드 둘레로 총 48개의 IO핀이 있고 VIN , GND, 5V, 3.3V 전원 연결핀이 있으며 USB 기기를 연결 할 수 있는 커넥터가 있습니다. 

어짜피 요요 보드의 펌웨어를 직접 변경하지 않으시고 사용하셔도 되므로 내장 MCU 칩이 AVR이건 PIC이건 상관없습니다.


요요 사용법

1. 전원
요요는 USB 호스트 역활을 수행하는 보드이며 연결된 안드로이드기기에 전원을 공급하여 충전할 수 있게 개발되어 있습니다.  즉, 자신뿐 아니라 연결된 안드로이드까지 커버할 정도의 충분한 전력이 필수적으로 요구됩니다.

VIN과 GND 단자에  전압이 7~12V정도의  *1A이상 전류 공급이 가능한 전원을 사용해야합니다.
정전압 5V 전원의 경우엔  VIN 단자 대신 5V 단자와 GND 사이에 직접 정전압 전원을 연결하셔도됩니다.  3.3V 단자에서는 레귤레이터로 감압된 3.3V 전원을 출력하여 활용할 수 있습니다.  (단, 3.3V단자에 3.3V 전원을 입력하는건 안됩니다)
관련 정보를 꼭 읽어보시고 연결하시기 바랍니다.

보드에 전원을 입력하면 POWER LED에 불이 들어옵니다.

2. Charge 전류량 조정용 트리머
요요에 전원을 연결후 안드로이드 기기와 연결해주면 안드로이드 기기의 충전이 시작되는 것을 확인할 수 있습니다.
하지만, 기기의 배터리 상태와 기기특성에 따라 충전과 요요 기기 인식이 안되는 경우가 있습니다. 이는 요요가 적절한 전력을 공급하지 못하는 경우인데요, 이때는 USB단자 옆에 있는 트리머를 시계방향으로 회전하여 전류공급량을 늘려주면됩니다. 기기마다 특성이 다른점에 대처하기 위해 본 트리머 설정기능을 제공하는것 같습니다.


3. 안드로이드 기기 설정

안드로이드 앱을 개발해오신 분들이라면 다 알고계신 내용입니다.

설정 > 응용프로그램 > 개발 > USB 디버깅  설정을 켜주셔야 합니다.


4. 프로그래밍

요요 보드 자체는 별도의 프로그래밍 작업이 필요없습니다.
안드로이드 기기에서 요요 보드에 있는 48개의 IO핀을  범용 디지털 입력,출력,아날로그입력(ADC),TWI, SPI,UART,PWM 등의 용도로 설정하고 값을 IO할 수 있는 라이브러리가 제공되기 때문입니다.   즉, 요요를 연결하면 안드로이드 기기에서 제어가능한 48개의 IO핀이 생겼다고 보시고 안드로이드 디바이스를 프로그래밍을 하시면 됩니다.

5. 케이블 연결 방법

케이블은 기존에 안드로이드 개발에 사용하셨던 기기용 USB 케이블을 그대로 사용하시면 됩니다.
PC와 안드로이드 디바이스를 USB 케이블로 연결 후 요요 연동용을 위한 프로그래밍을 하신 후,
PC쪽 단자를 뽑아서 요요에 연결하시면 됩니다.


6. 요요 보드 사용법

본래 요요보드는 Android 1.5 부터 지원되는 MicroBridge (ADB의 구현)를 활용하도록 개발되었으며 이는 구형 안드로이드OS(1.5이상) 에서도 활용이 가능한 장점이 있습니다.  하지만 최근에 새롭게 발표된 Open ADK (Android 2.3.4 이상 지원)의 지원도 가능하도록 새로움 펌웨어가 제공되고 있습니다. IOIO의 ADK 펌웨어는 현재 beta버젼이며 필요하신 경우 요요보드의 펌웨어를 변경후 사용가능한 방법입니다.

본 글에서는 펌웨어 변경없이 요요의 기본 연동방식을 이용합니다. 제가 갖고있는 안드로이드 기기가 1.5~2.2 정도만 지원되므로 당장 Open ADK 방식의 테스트는 어려울것 같습니다;;

7. ADB?  MicroBridge? 
개발시 디버그 용도로 사용되는 통신방법 및 툴 정도로 생각하면 될 것 같습니다. 이를 활용하여 안드로이드에 연결된 장치와 통신을 할 수 있습니다.  여러분도 보유하신 MCU보드에 USB호스트 기능만 추가하고 본 프로토콜 구현만 해주시면 안드로이드 연동 보드 개발이 가능합니다.  사실 Open ADK도 이것에서 파생되었다고 볼 수 있습니다.


8. 요요 IO 핀맵

요요 보드 뒷면에는 동그라미와 네모, 알바벳 P 등으로 각 핀의 기능을 간략히 표현하고 있습니다.  가령 네모 박스된 핀들은 모두 아날로그 입력핀으로 사용할 수 있으며,  동그라미 표시된 핀은 5V 입력이 허용되는 핀입니다.  요요의 기본 입출력 전압 레벨은 3.3V이지만 일부 핀의 경우 5V 입력이 허용되는 것 입니다. 자세한 핀별 용도는 별도의 상세페이지로 소개되고 있습니다. (IOIO핀맵 핀별 기능소개)


2부. 실습

이제는 직접 요요보드에 프로그래밍 후  기초적인 IO를 수행해 보도록 하겠습니다.
아래에 소개된 예제들은 안드로이드 프로그래밍이 처음인 분들을 위한 튜토리얼에 포함되 기본 예제와 아래의 IOIO 기본 공개 예제들을 그대로 사용하거나 몇 줄만 수정한 것입니다. 직접 해당 소스를 보시고 응용해보시기 바랍니다.
http://codaset.com/ytai/ioio/source/master/tree/software/applications

안드로이드 프로그래밍 자체가 처음이신 분들은 요요를 사용하시기 전에 우선 안드로이드 프로그래밍 기본예제를 꼭 실습 후 진행하시길 권장드립니다.  안드로이드  개발환경 설치 및 디바이스 인식 등이 마무리 되어야 요요 보드 응용 개발이 가능합니다.  거꾸로 마이크로 콘트롤러 보드와 같은 전자회로에 대한 경험이 없으신 분들은 Open ADK 보드의 기준 플래폼인 아두이노로 기초 학습을 진행하실것을 추천드립니다.


예제1-1. 상태 LED 제어하기

안드로이드 프로그래밍이 처음인 분들을 위한 튜토리얼에 포함된 기본 예제입니다.
보드에 내장된 stat(상태) LED를 On/Off 하는 예제이므로 별도의 전자회로를 연결할 필요가 없이 요요 보드에 전원만 공급해주면 테스트 해볼 수 있는 간단한 예제입니다. 


동영상1. 요요 기본사용방법과 예제1-1 수행 장면입니다.



예제1-2. 디지탈 출력으로 SSR제어하여 220V 전등 On/Off하기

예제1-1.의 소스에서 상태 LED를 제어하는 소스를 그대로 한줄 더 복사하여 특정 디지탈 출력핀을 On/Off하고, 이 출력 값을

SSR(Solid State Relay)

에 연결하여 가전용 전등을 On/Off 하는 예제입니다.


동영상2. 안드로이드로 220V 전등 On/Off 제어



예제2-1. 아날로그 입력 및 PWM 출력

IOIOSimpleApp

예제가 사용되었습니다.

가변저항으로 0~3.3V 범위의 임의 전압을 만들고 요요 보드의 ADC(아날로그 입력) 기능으로 입력받은 전압값을 수치화 하여 보여주는 예입니다.  더불어  디지탈 출력핀을 통해 PWM 파형을 출력하는 예제입니다.  기본 예제에서는 스크롤바의 위치에 대응하여 0.5ms ~ 2.5ms duty의 파형을 출력하도록 되어있으며 정확한 출력이 나오는지 확인을 위해 오실로스코프로 관측해봤습니다.

동영상3. PWM 출력과 아날로그 입력

예제2-2. PWM 출력으로 서보모터 제어

예제2-1. 에서 사용한 소스를 그대로 활용하여 RC서보모터를 구동할 수 있습니다.  다만 일부 서보모터의 회전각 범위는 0.5~2.5 ms 범위를 지원하지 않으므로 서보모터의 안전을 위해 이값을 0.8~2.2ms 범위로 수정하여 테스트 하였습니다.
이를 위해 PWM duty 관련 소스를 아래와 같이 수정하였습니다.

      pwmOutput_.setPulseWidth(800 + (int)(seekBar_.getProgress() * 1.4));


예제2-3. PWM 출력으로 전압메터 변화보기

이번 예제도 예제2의 소스를 그대로 사용하였습니다.   0~5V 범위의 전압을 측정할 수 있는 아날로그 전압 메타를 PWM 출력 핀에 연결하여 PWM duty 변화에 따른 출력 전압변화로 눈금의 변화를 확인해 보는 예제입니다.  기본 소스는 duty를 0~100%로 수정하면 약 0~3.3V 출력 변화를 확인 하실 수 있습니다.  전압 메타는 보유하신 멀티테스터 전압계를 이용하셔도 됩니다. PWM 파형의 Duty 비율을 조절하는것으로 서보모터 제어나  LED의 밝기 제어, 색상 제어 등 을 하실 수 있습니다.


동영상4. PWM제어로 RC서보모터 제어 및 전압메터 변화 관측

제품 링크

 

추가사항

IOIO 공식 자료 링크 변경 안내
 글 내용중 codaset 사이트에 보관되었던  모든 자료가 

https://github.com/ytai/ioio/wiki

로 이전되었으니 신규 사이트 정보를 이용해주시기 바랍니다.)

 

아두이노로 제어하기 좋은 로봇을 찾아오다 얼마전 드디어 쓸만한 녀석을 발견했습니다.

바로 무선제어 6족보행 로봇이란 제품입니다.





6족 보행로봇으로 전진,후진,좌회전,우회전이 가능하며 제공되는 적외선 리모컨을 통해 무선 조정이 가능합니다.
여러대의 로봇을 동시에 조정할 수 있도록 점퍼세팅으로 총 6대의 로봇을 동시에 조정할 수 있게 제작되었습니다.
조립도 간단하고 접착제같은건 사용하지 않아 신속하고 깔끔하게 제작이 가능합니다.

왜 좋은가?

1. 저렴한 가격
2. 손쉬운 제작 ( 8세이상 조립가능, 납땜불필요)
3. 손쉬운 아두이노 연동 가능( 전선 결합 불필요, 적외선 통신으로 제어)


조립시 필요한 것:
1.5V 알카리인 건전지 4개( 조정기용 2개, 로봇본체용 2개,  1.2V 충전지로도 잘 작동됩니다.)
스크류 드라이버

아두이노로 제어할 경우 필요한 재료:
 아두이노, IRED(적외선 LED) 1개,  저항 1개
(5미터 이상의 장거리 제어를 원하면 트랜지스터 1개, 저항 1개가 추가로 필요합니다.  관련글 링크 참고)


동영상1. 로봇 기본동작 보기

리모컨 제어로 전진, 후진, 좌회전, 우회전이 가능합니다.


아두이노로 제어하기

1. 하드웨어 준비
적외선 리모컨 신호를 아두이노 디지털 출력을 통해 출력하면됩니다.  이를 위해서 IRED(적외선LED) 1개와 저항 1개가 전부입니다.   원거리에서 제어하려면 트랜지스터를 사용해야 합니다.  연결방법은 일반 LED와 동일하게 저항1개와 함께 직렬 연결해주시면 됩니다.  본 예제에서는 PWM출력을 지원하는 아두이노 디지털 3번핀을 사용합니다.

리모콘 신호 전송을 위해 아두이노에 IRED 연결


IRED연결은 일반 LED 연결과 유사합니다.



2. 라이브러리 준비
적외선 리모컨의 역활을 아두이노가 대신하려면 여러분이 직접 38KHz의 캐리어 주파수에 수신기의 코드부호를 실어서 출력해주면 됩니다.  하지만 많은분들에게 쉽지 않은 도전과제이므로 천재적이면서 친절하기까지 하신 분들이 미리 만들어둔 라이브러리를 활용하여 비교적 손쉽게 아두이노로 리모컨 신호를 발신할 수 있습니다. 

아래의 링크에 가셔서 해당 라이브러리를 아두이노 스케치 library 폴더안에 복사하여 라이브러리 설치를 완료합니다.


위에서 준비해놓은 라이브러리는  RC5, SONY, NEC 코드 등의 송신 및 수신을 위한 기능이 제공되며 여기서는 NEC코드 발신 함수만 사용하면 됩니다. 리모컨 신호를 이용해 가전기기를 제어하는 것에 대해 일전에 소개해 드린적이 있는데요  이번에도 마찬가지로 코드부호를 스캔하여 해독한 결과 이 로봇도 NEC코드와 유사한 전송방식을 사용하는것을 알게되었습니다.
차이점은 데이타 전송폭이 짧다는 점과 반복표시용 코드를 사용하지 않는다는 것입니다. 

하지만 로봇제어를 위해 복잡한 내용은 모르셔도 상관 없습니다.  단지, 아래와 같은 함수 하나만 사용하기 때문입니다.


  irsend.sendNEC( codedata , nbits );   // codedata에 수신기용 코드값과   nbits에 코드값의 비트 수를 적어주면 됩니다.

위와 같은 함수로 해당코드값을 전송하면,  아두이노의경우 디지탈 3번핀에 해당되는 코드신호가 출력되고 이곳에 연결된 IRED를 통해 빛신호로 변환되어 출력되고  로봇에 내장된 적외선 수신기에서 해당 코드를 수신한뒤 대응되는 모터의 작동이 이뤄지게 되는 것입니다.


아래의 소스는 1초에 한번씩 전진, 후진, 좌회전, 우회전 신호를 발신하는 예제입니다.


#include <IRremote.h>

IRsend irsend;  

void setup()
{
// 특별한 초기화 설정이 불필요합니다.
}

void loop() {
   irsend.sendNEC( 0xc5000000, 8); //전진
   delay(1000);
   irsend.sendNEC( 0x45000000, 8); //후진
   delay(1000);
   irsend.sendNEC( 0x25000000, 8); //좌회전
   delay(1000);
   irsend.sendNEC( 0x85000000, 8); //우회전
   delay(1000);
  
}


위의 소스는 1초에 한번씩만 코드를 발송하므로 지속적으로 작동상태가 유지되지 않습니다. 따라서 만약 전진을 지속적으로 하려면 1초에 수회 정도 전진에 해당되는 코드를 반복하여 전송해야합니다. 


동영상2. 위 소스로 로봇 제어되는 장면

보통 이동형 로봇을 제작시 아두이노같은 마이크로콘트롤러 보드에 모터 구동 회로를 장착하고 모터 2개를 제어할 경우 방향제어를 위해 최소 4개의 전선을 연결하여 제어하게 됩니다. 이에 반해 본 로봇킷의 경우 모터 회로 구현이 이미 되어있는데다 전선의 연결이 불필요하고 빛으로 연결되어 있으므로 비용도 절감되고 제어도 간단해 질 수 있으므로 기초 이동형 로봇 제어 학습시 효과적으로 응용할 수 있는 방법으로 판단됩니다.  (물론 전용 모터드라이버 연결법이 더 다양한 제어가 가능합니다. 가령 속도조절이 가능함)

거리센서 등을 통해 주변 물체를 감지하고 회피하는 로봇의 제작이나  전파(Zigbee, Bluetooth, Wifi 등 활용)를 통한 무선통신으로 신호를 수신 받고 이를 광통신으로 변환하여 원격제어되는 로봇의 구현도 가능합니다.

게다가 수신모듈+모터드라이버 역활을 하는 제어기만 활용할 수도  있습니다.


사진. 리모컨 수신기 및 모터구동 드라이버 보드: 
좌측에 전원연결(2핀)부, 우측에 모터2개 연결부(4핀), 상부에 ON/OFF스위치,  하부에 리모컨 채널 조정용 헤더핀(6핀)이 있으며 가운데에 IR수광모듈이 보입니다.


기타.  작동중 오동작이 계속되면 건전지를 교체해 보시기 바랍니다.  모터 구동시 전력소모가 큰 편이므로 배터리 소모가 빨리 됩니다.


관련 제품 링크

 무선제어 6족보행 로봇

 아두이노 UNO

 적외선 무선통신킷



내용추가
2011.July.26 
아래의 관련글 링크에 리모콘 신호 송수신 회로와  PC에서 키보드 입력으로 로봇을 제어하는 소스 예제를 포함한 글을 새로 등록하였습니다. 참고하세요


관련글 링크

 .IR무선 리모콘 송수신 회로로 로봇 제어하기



로보밥 여름특집!!!   얼렁뚱땅 움직이면 쏘는 버블건을 제작했습니다.

아파트 계단이나 현관문에 많이사용되는 모션센서형 전등을 많이 보셨을 겁니다.   사람이 없을때는 미작동 되다가 다가오면 작동되므로 에너지 절약도 되고, 스위치 작동시키는 불편함도 없으므로 실생활에 매우 유용한 자동화 기기입니다.

이때 사용되는 센서가 PIR 근적외선 모션센서인데요 인체에서 방사되는 근적외선의 변화량을 감지하는 PIR센서가 들어있습니다. 아마도 많은 분들이 이를 응용하여 무언가를 만들고 싶어하실겁니다. 저도 마찬가지죠 ^^.   마침 며칠전 대형 할인점에 갔다가 버블건을 보고 좋은 응용예가 생각이 나서 만들어 봤습니다.

버블건의 원리는 모터의 회전시 물펌프로 비누액체를 뿜어올려주면서 동시에 팬으로 바람을 불어 비누방울을 날려주는 것입니다.  헉;;;  이 가격에 이런 기계를 만들어내는 분들에게 감사의 인사를 드리고 싶습니다 ^^.

모터 하나로 물펌프와 송풍용팬을 동시에 작동시키는 구조인데요, 보통 모터를 제어하려면 모터 드라이버를 사용하거나 릴레이를 사용하는게 정석일테지만,  이 버블건은 모터만 켜주면 되는건 아니고 방아쇠와 연결된 핸들이 비누액을 버블홀에 골고루 발라주는 역활을 하므로 반드시 방아쇠를 잡아 당겨줘야하는 구조였습니다.

음,,,   결국  서보모터 하나로 해결이 되었습니다.  그냥 손가락으로 당기듯이 서보모터를 작동시키는 방법입니다.
 단, 방아쇠의 스프링 장력이 좀 쌘것같아서 일부를 절단한 상태입니다.

버블건도 전등같이 하루 종일 켜져 있으면 효율적이지 않겠죠?!
항상 켜져있는것보다 사람이 곁에 있을경우에만 작동시킬 수 있으므로 효율적이며,  모르고 다가온 사람에게 놀라움을 선물할 수 도 있게됩니다. 


 

 

 





구현TIP
PIR 모션센서는 5V~12V에서 작동됩니다만,  9V이상의 전압에서 더 반응성이 좋습니다.  가령, 5V전원으로 사용시 감지된 뒤에 한동안 재 감지가 잘 되지 않는 현상이 있습니다.  9V전원을 사용후 이증상이 없어지는것을 확인했습니다.  즉, 수시로 감지되는 경우 9V이상의 전압으로 작동하시기 바랍니다.  단, 신호출력선은 아두이노에 입력해야 하므로 9V에 연결하시면 안되고 5V전원선과 10K옴 저항으로 연결해주셔야 합니다.


 

회로 연결 방법


아두이노 스케치 소스
#include <Servo.h>

const int ledPin = 13;
const int inputPin = 2;
const int posA = 90;
const int posB = 60;
Servo myservo;
unsigned long lastShotTime;

void setup(){
  pinMode(ledPin, OUTPUT);
  pinMode(inputPin, INPUT);
    myservo.attach(9);  // attaches the servo on pin 9 to the servo object
  lastShotTime = millis();
  Serial.begin(9600);
}

void loop(){
  int val = digitalRead(inputPin);
 // Serial.println( val);
   delay(500);
   if(val){
     myservo.write(posA);
     digitalWrite( ledPin, LOW);
   }else{
    digitalWrite( ledPin, HIGH);   
    trigger();
   }
}

void trigger(){
  if((lastShotTime + 20000UL) < millis() ){
    readyShot();
  }
  longShot();
  lastShotTime = millis();
  Serial.println( lastShotTime);
}

void readyShot(){
  myservo.write(posB);
  delay(500);
  myservo.write(posA);
  delay(500);
  myservo.write(posB);
  delay(500);
  myservo.write(posA);
  delay(500);
  myservo.write(posB);
  delay(500); 
  myservo.write(posA);
  delay(500);   
}

void longShot(){
  myservo.write(posB);
  delay(3000);
  myservo.write(posA);
}

소스 소개
PIR 센서에서 움직임이 감지되면, 출력선의 전압신호가 High에서 Low로 떨어집니다. 이를 디지탈핀에서 감지하여 서보모터를 회전시켜서 버블건을 작동시키게 됩니다.   지정된 시간(20000ms, 20초)동안 미작동상태에서 작동될 경우, 초기 거품 생성을 위해 3회를 짧게 작동시킨 후 지정된 시간(3000ms, 3초)동안 작동 되도록 프로그램 되었습니다.


재료:
1. 버블건 4,000원 ~ 10,000원 정도 (문방구, 대형할인점)
2. 아두이노(Arduino) 
3. PIR 근적외선 모션센서 
4. 서보모터
5-1. 휴대형 전원 => 9V 배럴잭 + 9V 건전지 (알카라인 고급형)
5-2. 실내형 전원 => 9V 아답터 전원 

( 9V 전지는 알카라인 고급형을 사용하셔야 잘 작동됩니다. 저가 9V 망간전지로는 서보모터 작동이 잘 안되네요)

 

연결법 안내 사진추가(2012.6.10)
안내해드린 모션센서가 5V에서 작동시엔 감이 너무 느려서 답답하며 9V ~12V 전원을 사용해야 정상 작동이 됩니다. 이때 어떻게 연결을 해야하는지 문의가 많으셨는데요...  늦게나마  아래의 사진을 올려드리오니 참고하시기 바랍니다. 아두이노 전원이 9~12V 정전압 전원인경우 아래 그림과 같이 아두이노에 있는 Vin 단자로부터 전원출력이 가능하므로 편리합니다. 별도의 전원을 사용하실때는  Vcc와 GND에 별도전원을 연결하시고  꼭  아두이노+센서+전원의 GND를 공통 함께 연결해주시기 바랍니다. 



 

아두이노에 WaveShield를 얹히면 음악재생기가 됩니다.  여기에 9V 전지등을 연결해주면 곧바로 휴대형이 되는데요, 무공해 에너지를 사용하면 더욱 좋겠죠?!  그래서 건전지 없이 태양전지를 연결하여 구동해 봤습니다.  쨍쨍 비치는 태양에너지를 쓸모있게 활용하는데 태양전지만한게 없겠죠...  날씨가 좋은경우 그늘에서도 문제없이 구동되네요.  생각과 달리 태양전지는 흐린날에도 약간이지만 전기를 생산합니다.  눈으로 한번 보시죠,  여러분의 창작품에도 응용하시면 좋을것 같습니다.

 

 

 

동영상1.음원은 개구리 우는소리로 넣었습니다.  개구리 우는 소리가 제법 큽니다.   확인을 위해 태양전지를 뒤집자 개구리가 울음을 끄치네요 ^^.   다시 태양을 바라보자 소리를 내기 시작합니다.


동영상2. 이제 제법 여름 기운이 느껴집니다.  너무 덥죠!  그늘에서도 작동하는지 테스트 해봤습니다.  잘~ 나오네요..  낮잠잘때 배경음으로 딱 좋을것 같습니다.  공원 밴치등에 불필요한 전력없이 태양전지로 작동되는  자연의 소리를 넣어주면 좋을것 같습니다.

동영상3.공원에서 되돌아 오는길에 자전거 뒷자석에 태양전지를 달고, 달려오면서 음악을 감상해 봤습니다.   제법 좋네요 ^^.
건전지 걱정없고,  스위치도 필요없습니다.  실내에 들어오면 자연히 꺼지고요,  밝은 곳으로 이동하면 다시 자동 재생됩니다.  와우~ Wow!!





^^. 로보밥 얼뚱연구소에 또하나의 재밌는 제품이 들어왔습니다. 예전부터 이제품 찾는 분들이 많았는데 드디어 입수하게 되었습니다.
 자세한 소개이전에 일단 테스트 영상을 올려봅니다.

Adafruit Industries사의 아두이노용 웨이브 쉴드이며,  직접 납땜하여 조립하는 DIY형 제품입니다.
mp3플레이어의 경우 전용 하드웨어 코덱칩을 사용하곤 하는데요, 요제품은 걍 wav파일을 곧바로 DAC칩으로 재생해버리므로 코덱같은거 없습니다.  단점일수있지만 반대로 장점일 수 도있죠,
low level 의 wav데이타를 곧바로 DA로 변형후 OPamp로 증폭하여 출력합니다.
그리고, 사실 mp3 배우기 전에  wav부터 배우는게 순서가 맞을것 같습니다.
SD메모리 슬롯도 내장하여 용량문제없이 음악재생이 가능하고, SD메모리만 응용할 수도 있습니다.

아래의 예제는 sd메모리에 미리 넣어둔 wav파일 몇개를 지정된 키를 누를때마다 재생하는 예제입니다.
관련예제는 해당제품 안내페이지에서 모두 보실 수 있습니다.
기회가 되면 상세 리뷰도 소개해 드리겠습니다.

 

 

 

 

 

PCB와 조립용 부품이 제공되며 직접 납땜으로 완성시키는 DIY형 제품입니다.

 

릴레이로 가전기기 제어하기

아두이노로 LED를 제어하여 ON/OFF 제어를 해보신 분들이라면 주변에서 널리 사용되고 있는 가전기기들도 ON/OFF 제어를 해보고 싶은 마음이 드실 겁니다. 이때 사용되는 전자부품으로 릴레이(Relay) 입니다.  보통 릴레이라고 하면 코일이 감겨진 전자석에 제어 전류를 흘려주면 그 자력으로 스위치의 접점부가 ON/OFF 되도록 설계된 기계식 릴레이가 많이 사용되고 있습니다.

 

 

사진.코일형 릴레이


조금? 다른 릴레이 SSR


오늘 실험에선 보통의 릴레이와 동일한 기능을 해주지만 좀더 사용하기 쉬운 SSR이라는 재밌는 전자부품을 사용하였습니다.
반도체 기술의 발달로 기계식 코일내장형 릴레이의 기능을 반도체 소자로 대체한 것이 SSR(Solid State Relay; 무접적 반도체 릴레이)입니다.  상대적으로 좀더 비싼편이지만 사용의 편리성과 안정된 성능으로 전기회로에 익숙치 않은분들이 가전기기의 제어용 릴레이로 사용하시기 좋은 부품입니다.

하지만, SSR도 용도별로 수많은 종류의 제품이 있어서 고르기가 쉽지만은 않습니다.
본 글에서는 4~32V의 직류 전원으로 240V 15A 까지의 교류 부하를 제어할 수 있으며, 납땜 작업 없이도 전선을 체결할 수 있는 볼트 결합단자가 제공되고, 보호 커버까지 장착된 SSR 제품을 선정하여 사용하였습니다.

 

 

 

사진. SSR 저전압(4~32V) 직류로 고전압 교류(240VAC) 부하를 제어하는 제품예

시중엔 이 제품보다 저렴한 제품이 많이 있지만, 일단은 사용하기 쉽고 편리한 제품의 사용을 추천드립니다.

SSR 의 특징
 .기계식 접점이 없으므로 접촉시 발생되는 노이즈가 없고 내구성이 우수합니다.
 .제품에 따라 매우 낮은 전압(가령 4~32V 직류)로 제어 할 수 있습니다.
 .코일형에 비해 상대적으로 매우 작은 전류만 소모합니다.
 .부하측 회로와 제어회로와의 우수한 절연성


릴레이 구동 회로

기계식 접촉식 릴레이를 사용할 경우 코일에 의한 역기전력이나 구동전류의 고려가 필요해지므로 제어측  회로가 좀더 복잡해집니다.  SSR의 경우 보통 입력신호측과 부하측이 전기적으로는 절연되어있고 광신호만 전달되므로 코일에 의한 역기전력이나 부하측과의 전기 절연에대해 고려할 점이 적어 회로가 매우 간단해 집니다.  6V건전지로 구동되는 스위치를 연결하거나 아두이노 같은 마이크로 콘트롤러의 디지탈 출력단자(5V출력)를 그대로 직결할 수 있습니다.


 

 

그림1. SSR로 전등 제어하는 초간단 회로도

주의 사항.
본 회로는 저전력 소모용 가전기기나 전등의 제어를 위한 것이며, 고전력 및 코일형 기기를 제어하는 경우 부가적인 안정화 회로가 필요할 수 있으며 고전류 구동시 방열을 위해 SSR의 밑면을 히트싱크(방열판)에 결합하여 사용해야 합니다. 또한, 전원 플러그와 소켓에는 안전을 위해 접지선까지 연결해 주실 것을 당부드립니다.


실험1. 6V건전지와 스위치로 220V 전등 ON/OFF하기

소개해 드렸듯이 입력측에 4~32V의 직류전압을 가하면 릴레이가 연결되고 교류회로가 작동되게 됩니다.  랜턴에 많이 사용되는 6V 건전지로 전등을 제어해 보는 장면입니다. 가장 단순한 회로의 예입니다.

실험2. 아두이노로 220V 전등 ON/OFF 제어하기

아두이노를 통한 제어도 원리상 동일합니다.  아두이노의 디지탈 출력핀에 High를 출력하면,  약 5V 전압이 출력됩니다. 그대로 SSR의 입력측에 연결해주면 LED를 ON/OFF하듯이 220V 전등이 ON/OFF 되게 됩니다.


너무 간단하죠 ^^.
하지만 220V는 간단치 않습니다. 잘못 만지면 크게 다치거나 돌아가실 수 있으니 주의해서 다뤄야합니다.
아래의 구성 예는 참고만 하시고, 스스로 충분히 안전에대한 준비가 되셨을때 스스로 충분히 검증을 하신 후 시도하시기 바랍니다.


주의사항 및 경고. 
본 글의 내용은 오류가 있을 수 있으며, 초보자들이 실습을 하기위해 필요한 모든 사전지식을 포함하고 있지 않습니다.
220V 교류 전원을 부주의하게 다루면 크게 다치거나 생명을 잃을 수 있습니다.  적절한 경험과 사전지식을 탐구한 이후에 주의하여 시도하시기 바랍니다.  여러분이 행한 실험에 대하여 저자는 어떠한 책임도 질 수 없습니다.


릴레이로 제어되는 콘센트 플러그 세트 만들기

이하, 위의 동영상에서 사용된 릴레이로 제어되는 콘센트 플러그를 만드는 방법을 소개해 드립니다.


 

 

1단계. 기본 재료:

플러그 암수 세트, SSR, 규격에 맞는 압착단자, 교류220V용 케이블(1.5mm 이상)

도구: 니퍼와 플라이어(뺀찌), 드라이버

 

 

 

플러그 부분에 부하선을 결합해 줍니다.

 

 

 

전선이 빠지지 않도록 매듭을 묶어주면 빠지지 않게됩니다. (매듭은 속에있어서 안보여요;;)

 

 

 

전용 압착단자를 이용하면 편리합니다.

 

 

 

 

 

 

플라이어로 꽉 눌러줘야 합니다.

 

 

 

 

 

 

 

암플러그 쪽에도 전선을 결합니다.

 

 

 

전선빠질 공간을 칼로 깍아냈습니다.

 

 

 

보통 잘라낼 수 있도록 미리 공간이 마련되어 있습니다.

 

 

 

이렇게 연결할까나~

 

 

 

 

 

요렇게 연결해 주시면 볼트가 풀려도 전선이 빠져나가는 사고는 미연에 방지할 수 있겠죠?!

 

 

 

회로도 보시면 어떻게 연결되는지 쉽게 아실 수 있으시죠?

 

 

 

건전지로 구동되는 아두이노라도 Okay입니다.

 

 

 

아답터 전원도 물론 좋구요

 

 

 

 

 

 

 

 

 

 

 

완성된 상태 여러장면들...


관련 제품 링크
 .무접접 반도체 릴레이 SSR
 .아두이노 Arduino
 .450/750V 기기배선용 절연선
 .전원플러그 220V형
 .전원 콘센트 1구형

추가 정보
아트로봇샵에서 취급중인 기계식 릴레이E형 제품의 사용법을 궁굼해하시는 분들이 계셔서 연결 예시 이미지를 추가로 올려봅니다.  예제에서는 12V전원에서 구동되는 모터를 아두이노에서 제어하기 위해서 릴레이E형과 아두이노를 연결한 모습입니다.

 

 

추가안내 사항.  2014.8.23일

전기기사님께서 댓글로 알려주신 정보입니다.  제가 예제소개시 사용한 전선의 색상(녹색)이 적절한 선택이 아니었던것 같습니다.  녹색 전선은 전기공사시 접지용도로 사용되어야하는 규칙이 있다고 하네요.  전기배선함이나  전등 스위치 단자를 열어보면 검정색선을 보통 사용하는것을 봤는데  나름대로 규칙이 있는것 같습니다. 이런 사항도 파악하셔서 꼭 지켜주시는것이 좋을것 같습니다.  좋은 정보 알려주신 익명의 전기기사님께 감사드립니다!!!

참고로 저전압을 사용하는 전자회로에서는 검정색은 GND 접지선으로 보통 사용됩니다.  고압을 사용하는 전기선과 서로 많이 다르네요 TT    이와 관련하여 어떤 역사적 스토리가 존재하는지 궁굼해집니다....   

 

 

 

오늘 소개해 드릴 녀석은 

EL 씨퀀서 아두이노 호환 보드(EL Sequencer)

  입니다.

일전에 소개해 드린(

인버터로 발광선 구동 안내기

) EL Wire 를 단순히 켜거나 점등하는 것으로는 부족하신 분들을 위한 시퀀서(제어/콘트롤) 보드입니다.
본 제품은 아두이노와 동일한 ATmega328칩을 내장하였고 아두이노 호환보드로 설계되어 일반 아두이노와 동일한 스케치(Sketch) 개발환경을 이용할 수 있으므로 비교적 손쉽게 EL Wire를 제어할 수 있게 해줍니다.  유사한 기능을 하지만 ATmega칩이 없이 아두이노 UNO에 장착하여 사용할 수 있는 쉴드형 제품(

아두이노용 EL제어 쉴드(EL Escudo)

도 있습니다.


EL Sequencer 프로그래밍 방법 소개

스케치에서 보드 선택은 "LilyPad Arduino w/ ATmega 328"  로 해주시면 됩니다.

아두이노 프로그래밍을 위해선 LilyPad나 Pro, Pro mini 와 동일한

FTDI USB시리얼 변환기

를 사용하시면 되며 5V형과 3.3V형 둘 다 사용 가능합니다.  FTDI보드 연결을 위한 6 Pin 핀홀에  일반 12mm 길이의 핀헤더나  90꺽인 핀 헤더를 납땜 후, FTDI보드를 연결해주시면 됩니다.   아래의 사진을 참고 하세요.

[사진. EL Sequencer 와 FTDI USB시리얼 변환 보드 연결방법]

참고로, 프로그래밍시엔 EL 인버터 전원을 꺼줄 것을 권장드리며,  전원 선택 스위치를 BAT가 아닌 USB 쪽으로 밀어주면, 보드 전원을 PC에 연결된 FTDI보드에서 공급받게 되므로 별도의 전원이나 배터리를 연결할 필요가 없습니다.

참고로, 총 8개의 EL Wire를 제어할 수 있으며 각각의 EL_Wire 단자마다 ON/OFF Relay역활을 해주는 TRIAC 소자를 이용해 인버터로 부터 나오는 교류 100~200V정도의 전원을 EL_Wire에 공급 여부를 선택하여 EL Wire를 ON/OFF 하게 됩니다.

프로그래밍 방법은 매우 간단하며, 아두이노를 처음 시작할때 배웠던 LED Blink 예제와 비슷한 수준입니다.
해당 디지탈 핀을 출력모드로 설정 후,
    pinMode( 2, OUTPUT);  // EL_A 단자 해당 디지탈핀(D2)를 출력모드로 설정

High 나 Low 신호를 출력하여 해당 디지탈핀의 출력을 제어하면, LED 가 ON/OFF 되듯이 EL_Wire가 ON/OFF 됩니다.

    digitalWrite( 2, LOW);  // EL_A 단자(D2)에 연결된 Wire 끄기
    digitalWrite( 2, HIGH);  // EL_A 단자(D2)에 연결된 Wire 켜기

가령 EL_A 단자에 연결된 EL Wire를 켜고자 한다면  아두이노 D2 핀 출력을 HIGH 로 설정해 주면 됩니다.

시퀀서 보드의 핀연결은 아래와 같이 되어 있습니다.

EL_단자와 대응되는 Arduino Digital Pin번호
 
Atmega328 핀번호
EL_A D2 PD2
EL_B D3 PD3
EL_C D4 PD4
EL_D D5 PD5
EL_E D6 PD6
EL_F D7 PD7
EL_G D8 PB0
EL_H D9 PB1

[표1. Pin Map]


매우 간단하고 응용하기 쉬운 예제 소스는 아래의 링크자료를 참고하시기 바랍니다.  튜토리얼 내용중에 GND끼리 납땜하는 작업은 안하셔도 됩니다. 패숑~드자이너 여러분께서는 튜토리얼 보시고 EL Wire를 활용한 패션쇼 한번 만들어보시기 바랍니다.  소스에 대해 몰라도 그냥 Copy/Paste 하셔서 Upload버튼만 눌러주시면 휘황찬란하게 8개의 EL Wire가 점등되는것을 보실 수 있습니다.

패션쇼 응용사례 소개 및 프로그래밍 소스예제 링크
 .

패션쇼용 EL Wire프로그래밍 따라잡기

( 출처: Makezine.com )
 .

위 튜토리얼에서 사용된 아두이노 소스 다운로드

EL Sequencer 프로그래밍 하기 동영상 소개

동영상1. 프로그래밍을 위한 FTDI 보드 연결방법

동영상2. 스케치 프로그래밍

동영상3. 인버터와 발광선 연결 후 실제 작동장면



부록. EL Wire 연결 방법

기본 장착된 EL Wire 연결 단자는 2pin JST 규격이므로  JST 2pin 커넥터가 연결된 EL Wire를 사용하면 그대로 연결 할 수 있습니다.  하지만, 모든 연결 단자 옆에 2pin 핀홀이 따로 있으므로 이곳에 원하는 규격의 단자를 연결하여 사용할 수 도 있습니다. 

 

 

[사진. JST 2pin 단자가 연결되어있는 EL Wire와 JST 2pin 케이블]

 

[사진. JST 규격 케이블과 JST Female 단자 ]


발광선(EL Wire)의 최대 장점중의 하나는 비교적 가공이 쉽다는 것입니다. 네온램프의 경우 일반인이 원하는 길이로 절단하고이를 원하는 모양으로 휘어서 가공하는 것이 가능할 지 모르겠습니다만, 발광선은 직접 원하는 크기로 잘라서 사용하실 수 도 있습니다. 이에관한 자료는 아래의 링크 정보를 참고하시기 바랍니다.

관련정보:

EL Wire 도움글:  http://www.ladyada.net/learn/el-wire/
EL와이어 작업방법: http://www.sparkfun.com/tutorials/130
EL wire 기타 예제:  http://www.digitalmisery.com/projects/halloween/el-ladder/

 

 

로보밥 얼뚱. 연구소에서 또 한가지 재밌고도 쓸만한 물건을 입수했습니다.
바로 RFID 스타터 킷 입니다.

 

 

 

 

 

 

 

[사진1. RFID 테스트 - 다양한 태그들]

RFID는 이미 버스, 전철, 신용카드, 아파트 출입구, 직원카드, 화물추적, 도서관 책, 상품 인식 및 보안용 태그 등으로 엄청난 양이 사용되고 있습니다.  수십년전 부터 많이 사용되오던 바코드와 유사하게 특정 값을 기억하고 손쉽게 리더기로 그 값을 읽어들여 신분확인이나 제품의 종류를 확인하는데 사용되고 있는데요  바코드가 아직은 더 저렴하고 일반적이지만 기억용량, 읽고 쓸 수 있는 능력, 자체 전원으로 원거리 인식 기능 제공등의 특수한 기능에 더 특화되어 응용될 수 있다는 장점으로 인해서 점점 더 영역을 넓혀 가고 있습니다.  참고로 최근의 최신 스마트폰에 내장된 NFC(Near Field Communication)도 전자기 유도현상이나 전자파를 통한 근거리 통신기술을 응용한 것으로 RFID와 유사한 기술의 하나로 볼 수 있습니다.

RFID는  주파수, 통신방식, 자체전원 여부 등에 따라 여러가지가 있습니다.
본 글에서는 가장 손쉽게 응용할 수 있는 읽기전용 카드 태그와 리더기 및 PC인터페이스 보드가 함께 제공되는 RFID 스타터킷에 대해 소개해 드리고자 합니다.

RFID 스타터킷의 구성품은 아래와 같습니다.
 .RFID 리더 모듈
 .PC USB통신용 FTDI 인터페이스 보드
 .카드형 RFID Tag(125KHZ형 ) 2개  [

제품링크

]

 

 

 

[사진2. RFID 모듈 뒷 모습 - 안테나 내장 ]

 

 

[사진 3. RFID모듈과 PC 인터페이스용 FTDI 보드 ]

관련 제품 링크
==

 

125KHZ형 RFID 태그와 리더기는 직원출입카드, 아파트 현관출입 카드 등으로 이미 많이 사용되고 있으므로 많은 경우 이미 보유하고 있는 카드를 그대로 이용하실 수 있습니다. 또한 RFID 리더 모듈은 안테나를 내장한 제품으로 아두이노 같은 손바닥 컴퓨터에 직접 연결하여 이동형 RFID리더 응용프로그램 개발이 가능하며, PC용 응용 프로그램을 제작할 때는 USB연결용 보드에 RFID 리더 모듈을 장착하면 PC측 USB포트에 연결하여 시리얼통신으로 RFID TAG의 코드를 인식하는 어플리케이션 개발도 가능하므로 활용성이 높습니다.


RFID 스타터킷 플래시 연동하기

플래시로 시리얼통신 장비와 연동을?

대부분의 시스템 프로그래밍 개발환경들은 시리얼 통신 제어가 가능하므로 RFID리더기로 부터 수신되는 데이타를 곧바로 처리할 수 있습니다. 하지만 아티스트 여러분들이 많이 사용하는 플래시는 아직 직접적인 시리얼통신을 할 수 없습니다.
때문에 시리얼통신을 플래시에서 지원되는 네트웍통신으로 중계해주는 프로그램을 사용해야 하는데요
가령 SerialProxy라는 프로그램이 있습니다.
본 글을 통해 실제 어떻게 프로그램들을 구성하는지와 네트웍 통신을 통해 수신되는 RFID 데이타를 어떻게 처리하여 응용할 수 있는지를 플래시용 예제 소스와 함께 소개해 드리고자 합니다.

일단, 예제 수행 동영상을 참고하시기 바랍니다.

[동영상1. 플래시 RFID 인식 시스템 구성 소개]

[동영상2. 플래시에서 RFID 인식하는 예제 수행]
태그 인식장면을 보기 쉽게 RFID리더를 모니터상단에 설치하였습니다.



RFID 스타터킷 + 플래시 인식 어플리케이션  준비 과정

1. RFID 설치
 RFID리더모듈을 PC연결용 보드에 장착후 USB케이블로 PC측에 연결해줍니다.
 이후, 운영체제가 장치를 인식하고 드라이버 자동설치를 시도하게 됩니다.
 운영체제에 따라 드라이버가 자동설치 될 수 도 있지만,
 자동인식이 안될 경우엔 FTDI 드라이버를 설치해 주시기 바랍니다.
 
 설치 완료후, 장치관리자>포트 리스트에서 해당 장치의 COM번호를 기억해 둡니다.
 
 간혹 케이블 연결 후 에도 장시간 아무런 반응이 없다고 문의주시는 분들이 있습니다.
 이경우, 케이블을 교체 후 연결해보시기 바랍니다.
 
2. SerialProxy 설치
시리얼프록시란 시리얼통신을 통한 데이타 입출력을 네트웍통신으로 중계해주는 프로그램입니다.

시리얼 통신만 가능한 장치도 이 프로그램을 이용하면 네트웍 통신이 가능해집니다.
반대로 네트웍통신만 지원하는 SW도 이 프로그램을 통해 시리엉 장치와 통신이 가능해집니다.
사람으로 치면 인터프리터(통역자) 정도 되는 녀석입니다.

다운로드하여 설치 후,
serproxy.cfg 파일을 수정하여 자신의 설정에 맞게 속도(9600), COM포트번호(가령 COM3)와 네트웍통신용 포트번호(가령 5331) 를 설정합니다.   또한가지 잊지말고  newlines_to_nils 가 false인지 확인해주시기 바랍니다.(false로 지정해야 불필요한 정보 전송이 안됩니다. true로 지정되면 플래시측 소스 수정요함)

 

설정파일 실제예

# Do not Transform newlines coming from the serial port into nils
newlines_to_nils=false

# Comm ports used
comm_ports=8

# Default settings
comm_baud=9600 
comm_databits=8
comm_stopbits=1
comm_parity=none
# Idle time out in seconds
timeout=300

# Settings for COM
net_port8=5331

 

 
 
3. 플래시 소스 수정하기
 플래시측 소스에서 SerialProxy와 통신을 위한 포트번호(가령 5331)를 확인하고 필요시 수정합니다.

4. SerialProxy 실행하기

5. 플래시 실행하기



RFID 내부 살펴보기

본 예제에서 사용된 태그의 경우 리더기로 값을 읽으면 태그 ID와 관련된 정보들이 Ascii코드로 시리얼 통신으로 전달됩니다.
아래와 같은 리더기 데이타 시트 정보를 참고하시면 해당 바이트의 의미를 파악할 수 있습니다.

 

 

 

[그림1. RFID리더기 데이타 출력 포멧 - 데이타 시트 참고 ]

 

리더기의 기본 출력 모드는 ASCII 코드 방식입니다.
가령 리더기에 태그를 인식시킬 때 마다 아래와 같이 16바이트 정보가 읽혀집니다.

02 34 35 30 30 42 38 45 45 43 42 44 38 0D 0A 03    //모두 16진수로 표기된 값입니다.

(위 정보는 시리얼 터미널 프로그램에서 시리얼 수신값을 16진수로 표시하면 볼 수 있으며,
아두이노 시리얼 모니터창으로 볼 경우 ASCII코드값으로 해석되어 다르게 출력되며 일부 문자가 깨져보이게 됩니다.)

16바이트의 정보는 각각 아래와 같이 해석할 수 있습니다.

0x02 // 1바이트 태그 정보의 시작을 알려주는 표시값 입니다.
0x34 // 10바이트 ASCII코드값의 시작
~     // 이 값들이 RFID의 고유 ID값 입니다.)
0x42 // 10바이트 ASCII코드값의 끝
0x44, 0x38  // 10바이트 id값의 2바이트 체크섬 (오류검사용, 보통 무시하거나 RFID ID값의 일부로 봐도 됩니다.)
0x0D // CR 캐리지리턴 (줄넘김 명령에 해당하는 ASCII 코드값)
0x0A // LF 라인피드(줄넘김 명령에 해당하는 ASCII 코드값)
0x03 // 태그 정보의 끝을 알려주는 표시값

위 정보를 시작 및 종료코드와 CR,LF 를 제외하고 ASCII코드로 표시하면 아래와 같습니다.

4500B8EECBD8

체크섬(D8)값을 제외하면 실질적인 태그의 고유 ID는 아래와 같이 표시 할 수 있습니다.

4500B8EECB
// ID값은 ASCII코드 중 숫자와 알파벳으로 표현 가능한 값으로 구성됩니다.

 

 

 

[그림2. 태그의 고유값 태그ID]

RFID 시스템에서는 위 ID값을 특정 사람이나 물건의 고유 ID값으로 설정해두고 리더기에서 위의 ID를 발견시 해당 사람이나 물건이 감지되었다고 인식하게 되는 것 입니다.

위의 플래시 예제에서도 미리 등록된 ID와 일치하는 경우 해당 이미지를 화면에 출력하였습니다. 만일 출입문에 응용된다면 출구를 열어주면되고, 도서관에서 도서대출시스템에 이용한다면 해당 ID의 책을 대출 처리하면 되는겁니다. 즉, 실제 응용방법은 바코드와 동일합니다.

본 스타터킷에서 사용되는 리더기와 태그는 태그의 ID값을 읽기만 가능하고 태그 ID는 미리 고정된 값으로 등록되어 있으므로 태그를 재프로그래밍할 필요가 없고 단순히 미리 지정된 태그값을 읽어서 일치하는 경우에 해당 업무를 수행하면 되므로 비교적 손쉽게 응용이 가능한 제품입니다.

스타터킷으로 손쉽게 다양한 RFID응용프로그램 개발을 경험해 보신 후, 좀더 본격적이고 복잡한 어플리케이션을 개발하실 경우 쓰기 지원 RFID시스템도 고려해 보시기 바랍니다.  추후 소개해 드릴 다른 종류의 RFID 모듈과 TAG의 경우 1KByte의 데이타를 태그에서 읽는 것은 물론이고 쓰기까지 가능한 제품도 있습니다.

 

 

 [사진4. 카드형 RFID Tag ]


 

 

[사진5. 유리캡슐형 RFID태크 - 동물의 체내 삽입용과 유사한 제품 ]
사람 몸에 넣으면 안되요;;


예제에서 사용된 플래시 소스를 첨부합니다. 

 

RFID-Tester.fla
다운로드

 

+ Recent posts