2014-12-11 16:13:49 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup driver_srf08
|
|
|
|
* @{
|
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2014-12-11 16:13:49 +01:00
|
|
|
* @brief Driver for the SRF08 ultrasonic ranger.
|
|
|
|
* The connection between the MCU and the SRF08 is based on the
|
|
|
|
* i2c-interface.
|
|
|
|
*
|
|
|
|
* @author Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
|
|
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "hwtimer.h"
|
|
|
|
#include "srf08.h"
|
|
|
|
#include "periph/i2c.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
|
|
|
|
int srf08_init(srf08_t *dev, i2c_t i2c, uint8_t addr, i2c_speed_t speed)
|
|
|
|
{
|
2015-01-20 08:54:40 +01:00
|
|
|
int status;
|
|
|
|
|
2014-12-11 16:13:49 +01:00
|
|
|
dev->i2c = i2c;
|
|
|
|
dev->addr = addr;
|
|
|
|
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-11 16:13:49 +01:00
|
|
|
/* initialize i2c interface */
|
2015-01-20 08:54:40 +01:00
|
|
|
status = i2c_init_master(dev->i2c, speed);
|
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
|
|
|
|
|
|
|
if(status < 0) {
|
2014-12-11 16:13:49 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-20 08:54:40 +01:00
|
|
|
|
2014-12-11 16:13:49 +01:00
|
|
|
/* set the maximum range */
|
|
|
|
if (srf08_set_max_range(dev, SRF08_MAX_RANGE_6M) < 0) {
|
|
|
|
return -3;
|
|
|
|
}
|
2015-01-20 08:54:40 +01:00
|
|
|
|
2014-12-11 16:13:49 +01:00
|
|
|
/* set the maximum gain */
|
|
|
|
if (srf08_set_max_gain(dev, SRF08_MAX_GAIN) < 0) {
|
|
|
|
return -4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int srf08_set_max_range(srf08_t *dev, uint8_t max_range)
|
|
|
|
{
|
2015-01-20 08:54:40 +01:00
|
|
|
int status;
|
|
|
|
|
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
|
|
|
status = i2c_write_reg(dev->i2c, dev->addr, SRF08_RANGE_REG, max_range);
|
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
|
|
|
|
|
|
|
return status;
|
2014-12-11 16:13:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int srf08_set_max_gain(srf08_t *dev, uint8_t gain)
|
|
|
|
{
|
2015-01-20 08:54:40 +01:00
|
|
|
int status;
|
|
|
|
|
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
|
|
|
status = i2c_write_reg(dev->i2c, dev->addr, SRF08_GAIN_REG, gain);
|
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
|
|
|
|
|
|
|
return status;
|
2014-12-11 16:13:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int srf08_get_distances(srf08_t *dev, uint16_t *range_array, int num_echos, srf08_mode_t ranging_mode)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
int echo_number = 0;
|
|
|
|
char range_bytes[sizeof(uint16_t)];
|
|
|
|
char register_location;
|
|
|
|
char max_reg_no_read = (num_echos * sizeof(range_bytes)) +1;
|
|
|
|
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-11 16:13:49 +01:00
|
|
|
/* set ranging mode */
|
|
|
|
status = i2c_write_reg(dev->i2c, dev->addr, SRF08_COMMAND_REG, ranging_mode);
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
2014-12-11 16:13:49 +01:00
|
|
|
|
|
|
|
if (!status) {
|
|
|
|
DEBUG("Write the ranging command to the i2c-interface is failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(max_reg_no_read > SRF08_MAX_REGISTER_NUMBER) {
|
|
|
|
DEBUG("Too many echos requested. Max. is 17");
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
hwtimer_wait(HWTIMER_TICKS(70000));
|
|
|
|
|
|
|
|
/* read all echo buffers */
|
|
|
|
for (register_location = 2; register_location < max_reg_no_read;
|
|
|
|
register_location += sizeof(range_bytes)) {
|
|
|
|
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Acquire exclusive access to the bus. */
|
|
|
|
i2c_acquire(dev->i2c);
|
2014-12-11 16:13:49 +01:00
|
|
|
/* read the echo bytes */
|
|
|
|
status = i2c_read_regs(dev->i2c, dev->addr, register_location, range_bytes, sizeof(range_bytes));
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
|
|
|
i2c_release(dev->i2c);
|
2014-12-11 16:13:49 +01:00
|
|
|
|
|
|
|
if (!status) {
|
|
|
|
DEBUG("Read the echo bytes from the i2c-interface is failed");
|
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (range_bytes == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
uint16_t distance = (range_bytes[0] << 8) | range_bytes[1];
|
|
|
|
range_array[(register_location - 2) / 2] = distance;
|
|
|
|
echo_number++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return echo_number;
|
|
|
|
}
|