The idea
The biggest problem I was having was that Brum didn’t detect all objects in front of it so it avoided some and ran straight in to others. This, I realised, was because I had misunderstood the detection area of the sensors. It’s ~30° in total and not ~60° as I believed, resulting in a to narrow detection field in front for it to notice obstacle on the sides(about wheel width). Making for a lot of crashing..
Also the “breaking”-feature I was using when switching between commands(drive forward, left, right etc) was useless and only resulting in lag and crashes. It didn’t break it just coasted.. So yeah, brain fart. Also the drive speed was way to fast!
This was indoors. Outdoors resulted in even more strange problems. The ultrasonic sensors don’t seem to like the cold and was giving no or lots of ghost results. Have been looking around for what the operational temperatures are for them but no luck, so if you know please enlighten me! Oh and on tarmac the drive speed was way to low due to the friction.
But this was results even if they wasn’t the results I was hoping for. Only thing to do now is to find solutions for them!
Below is a demo video of Brum running, avoiding an obstacle and crashing.. The code is also below, same as last post.
The Code
/**
* Self drivning RC-car code, Sensors: front, left front, right front
* Author: Henrik Norberg
* E-mail: henrik@dbhn.se
*/
/** LCD Setup **/
#include <Wire.h>
#include <LCD.h>
#include <LiquidCrystal_I2C.h>
#define I2C_ADDR 0x38 // <<----- Add your address here. Find it from I2C Scanner
#define BACKLIGHT_PIN 3
#define En_pin 2
#define Rw_pin 1
#define Rs_pin 0
#define D4_pin 4
#define D5_pin 5
#define D6_pin 6
#define D7_pin 7
LiquidCrystal_I2C lcd(I2C_ADDR,En_pin,Rw_pin,Rs_pin,D4_pin,D5_pin,D6_pin,D7_pin);
/** Engines **/
//Drive Engine
int motorAPin_A = 10; //Arduino digital 8 is connected to HG7881's A-1A terminal
int motorAPin_B = 11; //Arduino digital 9 is connected to HG7881's A-1B terminal
//Steering Engine
int motorBPin_A = 12; //Arduino digital 8 is connected to HG7881's B-1A terminal
int motorBPin_B = 13; //Arduino digital 9 is connected to HG7881's B-1B terminal
/** Ultrasonic sensors **/
// Ultrasonic Front
const int Trig0_pin = 1;
const int Echo0_pin = 2;
long duration0;
// Ultrasonic Front Left
const int Trig1_pin = 3; // pin for triggering pulse INPUT
const int Echo1_pin = 4; // pin for recieving echo OUPUT
long duration1; // how long it takes for the sound to rebound
// Ultrasonic Front Right
const int Trig2_pin = 5;
const int Echo2_pin = 6;
long duration2;
/** Misc settings **/
// Wall sensing or not (true or false for now)
boolean isFrontWall = false;
boolean isFrontLeftWall = false;
boolean isFrontRightWall = false;
//Starting values
int i = 150; // Initial engine speed
int motorState = 255; // Drive motor direction
void setup(){
lcd.begin (16,2); // <<----- My LCD was 16x2
// Switch on the backlight and print headline
lcd.setBacklightPin(BACKLIGHT_PIN,POSITIVE);
lcd.setBacklight(HIGH);
lcd.home (); // go home
lcd.print("Engine Test");
/**
* When program starts set Arduino pinmode for 10 to 13 digital to be OUTPUT
* so we can use analogWrite to output values from 0 to 255 (0-5V) (PWM)
*/
pinMode(motorAPin_A, OUTPUT); //direction
pinMode(motorAPin_B, OUTPUT); //speed
pinMode(motorBPin_A, OUTPUT); //direction
pinMode(motorBPin_B, OUTPUT); //speed
// ultrasonic Front
pinMode(Trig0_pin, OUTPUT); // initialize the pulse pin as output:
pinMode(Echo0_pin, INPUT); // initialize the echo_pin pin as an input:
// ultrasonic Front Left
pinMode(Trig1_pin, OUTPUT); // initialize the pulse pin as output:
pinMode(Echo1_pin, INPUT); // initialize the echo_pin pin as an input:
// ultrasonic Front Right
pinMode(Trig2_pin, OUTPUT); // initialize the pulse pin as output:
pinMode(Echo2_pin, INPUT); // initialize the echo_pin pin as an input:
}
void loop() {
checkFront();
if (isFrontWall == false) {
turnReset();
// Break if the car wasn't driving forward, saves on the engine and gears
if (motorState != 255) {
lcd.setCursor (0,1);
lcd.print("Drive - Stop ");
analogWrite(motorAPin_A, LOW);
analogWrite(motorAPin_B, LOW);
delay(1000);
analogWrite(motorAPin_A, 255); //Drive forward
motorState = 255;
i = 150;
}
else {
analogWrite(motorAPin_A, 255); //Drive forward
motorState = 255;
lcd.setCursor (0,1);
lcd.print("Drive - Forward");
if (i << 180){ // Speed up if not at max speed
for(i;i<=180; i++){
analogWrite(motorAPin_B, invertOurValue( i ));
delay(40);
}
}
analogWrite(motorAPin_B, invertOurValue( i ));
}
}
else {
checkFrontLeft();
checkFrontRight();
if (isFrontRightWall == false && isFrontLeftWall == false) {
//Sides are free, turn left
analogWrite(motorBPin_B, 255 );
// Break if the car was driving forward or reversing
if (motorState != 1) {
lcd.setCursor (0,1);
lcd.print("Drive - Stop ");
analogWrite(motorAPin_A, LOW);
analogWrite(motorAPin_B, LOW);
delay(1000);
i = 150;
analogWrite(motorAPin_A, 255); //Drive forward
motorState = 1;
}
else {
analogWrite(motorAPin_A, 255); //Drive forward
analogWrite(motorBPin_A, 0); //Turn left
motorState = 1;
lcd.setCursor (0,1);
lcd.print("Drive - Left ");
if (i << 180){ // Speed up if not at max speed
for(i; i<=180; i++){
analogWrite(motorAPin_B, i);
delay(40);
}
}
analogWrite(motorAPin_B, i);
delay(500);
}
}
else {
if (isFrontRightWall == true && isFrontLeftWall == true){
// Break if the car wasn't reversing, saves on the engine and gears
turnReset();
if (motorState != 0) {
lcd.setCursor (0,1);
lcd.print("Drive - Stop ");
analogWrite(motorAPin_A, LOW);
analogWrite(motorAPin_B, LOW);
delay(1000);
i = 150;
analogWrite(motorAPin_A, 0); //Drive backward
motorState = 0;
}
else {
analogWrite(motorAPin_A, 0); //Drive backward
motorState = 0;
lcd.setCursor (0,1);
lcd.print("Drive - Reverse");
if (i << 180){ // Speed up if not at max speed
for(i; i<=180; i++){
analogWrite(motorAPin_B, i);
delay(40);
}
}
analogWrite(motorAPin_B, i);
delay(500);
}
}
else {
if (isFrontRightWall == true && isFrontLeftWall == false) {
//Left is free, Right is blocked. Turn Left and go forward
analogWrite(motorBPin_B, 255 );
// Break if the car was driving forward or reversing
if (motorState != 1) {
lcd.setCursor (0,1);
lcd.print("Drive - Stop ");
analogWrite(motorAPin_A, LOW);
analogWrite(motorAPin_B, LOW);
delay(1000);
i = 150;
analogWrite(motorAPin_A, 255); //Drive forward
motorState = 1;
}
else {
analogWrite(motorAPin_A, 255); //Drive forward
analogWrite(motorBPin_A, 0); //Turn left
motorState = 1;
lcd.setCursor (0,1);
lcd.print("Drive - Left ");
if (i << 180){ // Speed up if not at max speed
for(i; i<=180; i++){
analogWrite(motorAPin_B, i);
delay(40);
}
}
analogWrite(motorAPin_B, i);
delay(500);
}
}
else
if (isFrontRightWall == false && isFrontLeftWall == true) {
//Right is free, Left is blocked. Turn Right and go forward
analogWrite(motorBPin_B, 0 );
// Break if the car was driving forward or reversing
if (motorState != 2) {
lcd.setCursor (0,1);
lcd.print("Drive - Stop ");
analogWrite(motorAPin_A, LOW);
analogWrite(motorAPin_B, LOW);
delay(1000);
i = 150;
analogWrite(motorAPin_A, 255); //Drive forward
motorState = 2;
}
else {
analogWrite(motorAPin_A, 255); //Drive forward
analogWrite(motorBPin_A, 255); //Turn Right
motorState = 2;
lcd.setCursor (0,1);
lcd.print("Drive - Right ");
if (i << 180){ // Speed up if not at max speed
for(i; i<=180; i++){
analogWrite(motorAPin_B, i);
delay(40);
}
}
analogWrite(motorAPin_B, i);
delay(500);
}
}
}
}
}
}
//Invert speed value
int invertOurValue(int input) {
return 255 - input;
}
//Checking for obstacle in front of car
void checkFront() {
digitalWrite(Trig0_pin, LOW);
delayMicroseconds(2);
digitalWrite(Trig0_pin, HIGH);
delayMicroseconds(10);
digitalWrite(Trig0_pin, LOW);
duration0 = pulseIn(Echo0_pin,10);
if ((duration0 > 4000 || duration0 == 0)) {
isFrontWall = false;
}
else {
isFrontWall = true;
}
}
void checkFrontLeft() {
digitalWrite(Trig1_pin, LOW);
delayMicroseconds(2);
digitalWrite(Trig1_pin, HIGH);
delayMicroseconds(10);
digitalWrite(Trig1_pin, LOW);
duration1 = pulseIn(Echo1_pin,10);
if ((duration1 > 4000 || duration1 == 0)) {
isFrontLeftWall = false;
}
else {
isFrontLeftWall = true;
}
}
void checkFrontRight () {
digitalWrite(Trig2_pin, LOW);
delayMicroseconds(2);
digitalWrite(Trig2_pin, HIGH);
delayMicroseconds(10);
digitalWrite(Trig2_pin, LOW);
duration2 = pulseIn(Echo2_pin,10);
if ((duration2 > 4000 || duration2 == 0)) {
isFrontRightWall = false;
}
else {
isFrontRightWall = true;
}
}
void turnReset() {
analogWrite(motorBPin_A, LOW );
analogWrite(motorBPin_B, LOW );
}
The Future
The future is actually already here but want to make one more video before showing it. Any how, Brum had to be totally reworked giving it wider detection range etc. Also A LOT of new code was needed but it have been worth the work so stay tuned!