Pages

Wednesday, 11 June 2014

Practice - Remotely controlling a car with Arduino by using Xbee

About

In this post we explain the process as usual. However, if you prefer it, you can download our class presentation slides by clicking on the dropbox link below:

Description

This time we are going to show you our final project for the Wireless Sensor Networks course, which consists in a remote controlled car built with an Arduino.

Communication between the remote control and the car is going to be supported with XBee's. As usual, here is the list of the needed material to the final build:
  • Hookup wires
  • 1x Arduino Uno
  • 2x XBee (with XBee explorer)
  • 2x BC547 Transistor 
  • 2x 1N4007 Diodes
  • 2x 1KOhm Resistors
  • 3x 10KOhm Resistors
  • 3x Push Buttons
  • 6x 1,5V Batteries (2x AAA + 4 x AA)
  • 2x  Battery Holders (one with capcacity 2 and the other with cap. 4)
  • 2x Motors
  • 3x Wheels  (one for support)
  • ... 
Before showing the circuit and the code used, we want to explain you some features about the electronics that we have applied on the design, which will roughly give you the needed knowledge:

Voltage batteries

When using circuits with different voltages, you have to connect their grounds together to provide a common ground. In our case, we have used 3V(2x 1,5V) and 6V (4x 1,5V) batteries for feeding one XBee and the motors, respectively.

Transistor

A transistor is an electronic device that can be used on circuits for amplifying or switching purposes. The transistor that we have used (of type Negative-Positive-Negative, NPN) is bipolar junction type and has three terminals (base, collector and emitter), which currents and voltages depend on the others, i.e. current on the base terminal controls the current on collector and emitter.

In our case, we have used transistors as switches, changing their state from OFF to ON as desired, in order to have control on the power supply. This can be done because transistors do not let the current path unless a saturation voltage on the Collector pin is reached.

Figure : NPN Transistor Scheme [1]
When maximum Collector current flows the transistor is said to be Saturated. The value of the Base resistor determines how much input voltage is required and corresponding Base current to switch the transistor fully “ON”. In our case, we had to implement a voltage divisor in order to get an input base intensity low enough to make the transistor act as a digital switch.

Figure : BC547 Transistor [2]

Diode

Diode is a polarized component which can go only one way in the circuit (it let the current path just in one direction) and it has a stripe on one end. That end is the negative end, or cathode, of the diode. The other end is the positive end, or anode. The anode of the diode is connected to the ground of the motor and the cathode of the diode to the power of the motor.

This may seem backward, and in fact, it is. The diode will help prevent any back-voltage generated by the motor from going back into the circuit. Back voltage will flow in the opposite direction of the voltage supply (must be avoided). The diodes we have used are Diodes 1N4007.

Figure: Diode 1N4007

Assembly of the pieces

To mount the car we used the breadboard as the support of all the materials (e.g. XBee Explorer, hookup wires, batteries, motors). We needed some way to stick all these materials; therefore we used some adhesive tape along with mini-boards of cork so that motors and batteries could be well fixed to the breadboard.

Figure: Car ready to be build (left circuit is the controller)
Figure: Put the motors over the first cork layer and stick them with adhesive tape (I)
Figure: Put the motors over the first cork layer and stick them with adhesive tape (II)

Figure: Cork layer used to mount the car
Figure: Stick the batteries to the last cork layer (there are 3 layers)

Figure: Final circuit (without battery feeding the XBee)

Figure: Final circuit (with battery feeding the XBee)
Moreover, we had problems with motors and wheels since the axis of the motors was not wide enough to be put in the wheel. To solve this we used some elastic weave put along the motor axis and then we put some adhesive tape over it. This made enough space for the wheel to fit well in the axis. Another possibility, which could work better, consists in adding blu-tack in the joint between the wheel and the axis. In any case, try to get a wheel that fits perfectly the motor axis to avoid these shoddy works.

Figure: Motor axis covered by elastic weave to fit into the wheel hole

Respect to the motors, it was necessary to weld them to hookup wires since it was not done. The motors we used are the ones at [3].

Figure: Welding the hookup wires to the motor (I)

Figure: Welding the hookup wires to the motor

There is also a support wheel which is not actioned by a motor. We also stuck to the last cork layer using blu-tack. It made that the height of the breadboard increased and it made not work completely well; ideally, it would have been put such that the height of the breadboard was completely straight.

Figure: Support wheel stuck with blu-tack

Finally, after adding all these components we have the final result:

Figure: Final car with the remote control to drive it

Circuits

Figure: Controller circuit
Figure: Car circuit


Code


  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
/**
 * This code is used to control the remote car by sending
 * messages through an XBee.
 */

// Pins of the Arduino to which the switches to move the car are connected
const int LEFT_PIN = 2;
const int FORW_PIN = 3;
const int RIGHT_PIN = 4;

// Pins to which we will transmit the information we want to send to the XBee
const int D1_PIN = 8; // corresponds to right wheel
const int D2_PIN = 9; // corresponds to left wheel

// Keeps track of the desired remote on/off state (left, right, forward)
int lRI = false; 
int rRI = false;
int fRI = false;

// Keeps track of the previous remote on/off state
int last_lRI= false;
int last_rRI = false;
int last_fRI = false;

// records last time the remote was re-set to keep it in sync
unsigned long lastSent = 0;

void setup()
{  
  // declare the switch (directions) pins as inputs   
  pinMode(LEFT_PIN, INPUT);
  pinMode(FORW_PIN, INPUT);
  pinMode(RIGHT_PIN, INPUT);
  Serial.begin(9600);
}

void loop()
{
  setRemoteIndicators();
  sendInformation();
}

// Read the value of the switches
//   digitalRead() checks to see if there is voltage on the pin or not  
void setRemoteIndicators()
{
  if (digitalRead(LEFT_PIN) == HIGH) // move left
  {
    lRI = true;
    rRI = false;
    fRI = false;
  } 
  else if (digitalRead(FORW_PIN) == HIGH) // move forward
  {
    lRI = false;
    rRI = false;
    fRI = true;
  }
  else if (digitalRead(RIGHT_PIN) == HIGH) // move right
  {
    lRI = false;
    rRI = true;
    fRI = false;
  }
  else // do not move
  {
    lRI = false;
    rRI = false;
    fRI = false;
  }
}

void sendInformation()
{
  // set the indicators immediately when there's a state change
  if (lRI != last_lRI)
  {
    if (!lRI) setRemoteState(0x4,'1');
    if (lRI) setRemoteState(0x5,'1');
    last_lRI = lRI;
  }
  delay(10);
  
  if (rRI != last_rRI)
  {
    if (!rRI) setRemoteState(0x4,'2');
    if (rRI) setRemoteState(0x5,'2');
    last_rRI = rRI;
  }
  delay(10);
  
  if (fRI != last_fRI)
  {
    if (!fRI)
    {
      setRemoteState(0x4,'1');
      setRemoteState(0x4,'2');
    }
    if (fRI)
    {
      setRemoteState(0x5,'1');
      setRemoteState(0x5,'2');
    }
    last_fRI = fRI;
  }

  // re-set the indicator occasionally in case it's out of sync
  if (millis() - lastSent > 10000 )
  {
    if (!lRI) setRemoteState(0x4,'1');
    if (lRI) setRemoteState(0x5,'1');
    delay(10);
    
    if (!rRI) setRemoteState(0x4,'2');
    if (rRI) setRemoteState(0x5,'2');
    lastSent = millis();
  }
}

// Send msg through the xbee (note that the frame has to be formed)
void setRemoteState(int value, char pinNum)
{
  Serial.write(0x7e);//Start byte
  Serial.write((byte)0x0);//Length
  Serial.write(0x10);//Length High
  Serial.write(0x17);//AT Command Request
  Serial.write((byte)0x0);//Frame ID
  Serial.write((byte)0x0);//Serial Number of Destination
  Serial.write((byte)0x0);
  Serial.write((byte)0x0);
  Serial.write((byte)0x0);
  Serial.write((byte)0x0);
  Serial.write((byte)0x0);
  Serial.write(0xFF);
  Serial.write(0xFF);//End of Serial Number of Destinition

  // 16 bit of recipient or 0xFFFE if unknown
  Serial.write(0xFF);
  Serial.write(0xFE);
  Serial.write(0x02);//Apply changes immediately

  //Command name in ASCII characters
  Serial.write('D');
  Serial.write(pinNum);

  //command data in as many bytes after length bytes
  Serial.write(value);

  //checksum is all bytes after length bytes
  long sum = 0x17 + 0xFF+ 0xFF + 0xFF + 0xFE + 0x02 + 'D' + pinNum + value;
  Serial.write (0xFF - ( sum & 0xFF) );
  delay(10); // avoiding overwhelming
} 

Video

References

[1] www.electronics-tutorials.ws/transistor/tran_2.html
[2] www.bhashatech.com/transistors/48-npn-bc547.html
[3] http://www.bricogeek.com/shop/motores/220-motor-micro-metal-dc-con-reductora-298-1.html

Tuesday, 3 June 2014

Practice - Measuring Light Pollution in Barcelona

Description

This time we are going to measure some values with sensors and send them automatically to a Visualization Platform (xively.com) through automatically generated json code.
The first part consits in a brief explanation about how to configure the system and how to send our first measurements.
Finally, we are going to extend all this with the aim to provide real evidences of how light pollution affects us and our environment (animals, plants...). As usual, we now provide a list of the material need for this practice, including HW and SW.

The materials we will need are the following:
  • Hookup wires
  • 2x XBee (1 configured as ZigBee Router API, 1 configured as ZigBee Coordinator API)
  • 2x XBee Explorer
  • 2x resistors of 22 kOhms (one is for backup)
  • 2x LDR
  • 1x Arduino Uno (to feed one of the XBees)
  • 1x Computer

The software you will need is the following:

Part 1

As in previous lab, we have an XBee set to ZigBee Router API and the other XBee set to ZigBee Coordinator API.

The router has to contain same PAN ID as coordinator and its address in the destination field. The field AP must be set to 2 (API enable) and we also have to configure port DO into ADC in order to receive data from the sensor.

Note that we are setting the IO Sampling Rate with 400 ms, which means that we send packets with rate 2.5 packets/second. We did that in order not to collapse Xively when we upload data.

Before sending data to Xively we must ensure that we have available "curl" command to allow communication with the Internet. If you are on windows you should download an executable and add it to your path (if you are working with cmd). You can access your computer's path variable by accessing environment variables on properties menu.

Another issue you must take into account before executing the code is to set the API endpoint with "https" or "http". It is possible that you find troubles with certifieds using "https", so we recomend you to use "http".

Now we are able to execute the code! If did not have already created an account on Xivel, you can access their webpage and create one. You will also need to add a new device and provide a description of it in order to obtain the "API Endpoint" and the "API Key", which must be included in your python code.

As the image below shows, we have uploaded our measurements into the Xively platform:

Figure 1: Measurements on Xively

Circuit Part 1

Figure 2: Picture Circuit Part 1
Figure 3: Circuit Part 1

Code Part 1 [3]

The following code reads incoming values of light measurements done on the coordinator side, creates a JSON file and executes the command for uploading this file to Xively platform. A new JSON file is  done and upload every time we receive data.

# Derived from code by Alejandro Andreu
import commands
import json
import serial
import time
import os
from serial import SerialException
from xbee import ZigBee

print 'Asynchronously printing data from remote XBee'

serial_port = serial.Serial('COM5', 9600)

def print_data(data):
    """
    This method is called whenever data is received.
    Its only argument is the data within the frame.
    """
    print data['samples'][0].keys()[0]

    # Create a JSON file and fill it with the received samples
    json_data = {}
    json_data['version'] = '0.2'
    json_data['datastreams'] = ()
    json_data['datastreams'] = json_data['datastreams'] + ({'id': data['samples'][0].keys()[0], 'current_value': str(data['samples'][0].values()[0])},)
    # Add more datastreams if needed
    with open('Xbee2Xively.json', mode='w') as f:
        json.dump(json_data, f, indent = 4)
    # Upload information to Xively. Use your own Api Key and feed identifier
    os.popen('curl --request PUT --data-binary @Xbee2Xively.json --header "X-ApiKey:g14aNzsNGLD0UOvi4wFXpTjzqfliK0JcfgpU0GnubyGUBzKi" --verbose https://api.xively.com/v2/feeds/779242912')

zigbee = ZigBee(serial_port, escaped = True, callback = print_data)

time.sleep(200)

zigbee.halt();
serial_port.close()

Part 2

Introduction/About

In this part we are going to experiment how light pollution affects Barcelona. About light pollution, so far, there is little social conscience, even though it generates numerous harmful consequences such as energy waste (and greenhouse gas emissions resulting from their production). This also effects to human and animals health and psychology, and, even some people could find not a priority, it also damages astronomy activity and generates general loss of perception of the Universe [1]. In fact, it has been shown in recent years that prolonged exposure of the trees to artificial light can cause the trees out of control and produce oxygen at night instead of absorbing carbon dioxide (which is one of the main purposes of plant trees in the city). Hence, we would like to contribute a little by making this project and divulge a bit the problematic.

As obvious, we consider that nowadays it is a must to have well illuminated streets at nights, but it should be a must to provide this light in an adequate, accurate and efficient way. As an example,  all sent laterally light, upward or to paces where it is not needed, does not provide security and visibility and is a waste of energy (environment harm) and money (we pay lot of taxes and the contributors have the right to make our politicians to invest where necessary).

Therefore, we can get some experimental data and go to city hall and force them to:
  • Install streets bulbs horizontally. Use shielded lamps and luminous flux directed only downward.
  • Install preferably sodium vapor lamps of low pressure and adequate power to use.
  • Turn off Monumental and advertising lighting after midnight (there will not be any citizen or tourist contemplating any monument).
  • Prohibit light or laser guns, and any projector that directs light toward the sky.
  • Use asymmetrical floodlights without inclination.
  • Reduce consumption in peak hours, at dawn, by brownout on the public or the selective off lights.


In short, the only way to control light pollution is to reduce the amount of light sent to heaven and reduce general consumption. We have to respect the night ecosystem.

Figure 4: Spanish night map [5]

Getting experimental data

To do this, we are going to use the same procedure as before but we are also adding another sensor for backup. The information of this last sensor would be included on the same json file that we sended before in order to compare the uploaded values on the Xively platform.

The previous IO Sampling Rate was 400 ms, but now we have twice amount of data, so we set this field on 1000 ms, that means 0.5 packet per second (we have 2 sources).

As we do not have enough sensor's precision, we are just going to compare the results obtained of measuring during the day and the night with the theoretical  illuminance values, which are described below:

Table 1: Illuminance values [4]
We can see that the equivalence on day-night values must be 120000:1 in the extreme cases. In our case, we obtain measurements on a cloudy day with shadows, so our equivalence should be 500:1.

The following pictures shows the measurements done by both sensors, which are done with the data received from Xively using the following cmd command:

curl --request GET http://api.xively.com/v2/feeds/YOUR_FEED/datastreams/YOUR_CHANNEL.csv?key=YOUR_API_KEY&start=2013-12-22&duration=24hours&interval=900&limit=100&interval_type=discrete" > data.csv

Figure 5: ADC-0 measurements


Figure 6: ADC-1 measurements

We can observe that measured light on the night is not far enough from received during a shinny day, so we can observe an equivalence of 4:1, which is so far away of our assumptions.
Note that both graphs show the same data from 23:00 of the first day to 0.00 of the second day. This is because Xively gaves us a graphic with the last token value (at 23:30) extended until the moment we requested the data.

Circuit Part 2

Figure 7: Picture Circuit Part 2
Figure 8: Circuit Part 2

Code Part 2 [3]

# Derived from code by Alejandro Andreu
import commands
import json
import serial
import time
import os
from serial import SerialException
from xbee import ZigBee

print 'Asynchronously printing data from remote XBee'

serial_port = serial.Serial('COM5', 9600)

def print_data(data):
    """
    This method is called whenever data is received.
    Its only argument is the data within the frame.
    """
    print data['samples'][0].keys()[0]
    print data['samples'][0].values()[0]
    print data['samples'][0].keys()[1]
    print data['samples'][0].values()[1]
    
    # Create a JSON file and fill it with the received samples
    json_data = {}
    json_data['version'] = '0.2'
    json_data['datastreams'] = ()
    json_data['datastreams'] = json_data['datastreams'] + ({'id': data['samples'][0].keys()[0], 'current_value': str(data['samples'][0].values()[0])},)
 
    with open('Xbee2Xively.json', mode='w') as f:
        json.dump(json_data, f, indent = 4)
  
 # Add data from adc-1 pin
    json_data1 = {}
    json_data1['version'] = '0.2'
    json_data1['datastreams'] = ()
    json_data1['datastreams'] = json_data['datastreams'] + ({'id': data['samples'][0].keys()[1], 'current_value': str(data['samples'][0].values()[1])},)
 
    with open('Xbee2Xively.json', mode='w') as f:
        json.dump(json_data1, f, indent = 4)
   
 
zigbee = ZigBee(serial_port, callback = print_data)
while True:
    try:
        time.sleep(500)
    except KeyboardInterrupt:
        break

zigbee.halt();
serial_port.close()

References

[1] Python software and libraries, retrieved from: https://www.python.org/
[2] XBee library for Arduino, retrieved from: https://code.google.com/p/xbee-arduino/
[3] Version of the code "upload_to_xively.py" from Jaume Barceló and Luis Sanabria
[4] Light pollution information from "Universidad de Murcia" report, retrieved from:  http://www.um.es/cieloscuro/documentos/divulgacion/cluminica_dos_palabras.pdf
[5] Image from "celfosc", retrieved from: http://www.celfosc.org/
[6] Illuminance information from Wikipedia, retrieved from: http://en.wikipedia.org/wiki/Daylight