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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-08-29 18:00:46 +02:00
|
|
|
* @ingroup drivers_srf08
|
2014-12-11 16:13:49 +01:00
|
|
|
* @{
|
|
|
|
*
|
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>
|
2018-06-27 08:25:51 +02:00
|
|
|
* @author Kevin Weiss <kevin.weiss@haw-hamburg.de>
|
2014-12-11 16:13:49 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
2020-02-21 12:05:15 +01:00
|
|
|
|
2015-08-14 10:22:46 +02:00
|
|
|
#include "xtimer.h"
|
2014-12-11 16:13:49 +01:00
|
|
|
#include "periph/i2c.h"
|
|
|
|
|
2020-02-21 12:05:15 +01:00
|
|
|
#include "srf08.h"
|
|
|
|
#include "srf08_params.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2014-12-11 16:13:49 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2020-02-21 12:05:15 +01:00
|
|
|
#define SRF08_DEV_I2C (dev->params.i2c)
|
|
|
|
#define SRF08_DEV_ADDR (dev->params.addr)
|
2014-12-11 16:13:49 +01:00
|
|
|
|
2020-02-21 12:05:15 +01:00
|
|
|
int srf08_init(srf08_t *dev, const srf08_params_t *params)
|
2014-12-11 16:13:49 +01:00
|
|
|
{
|
2020-02-21 12:05:15 +01:00
|
|
|
dev->params = *params;
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int srf08_set_max_range(const srf08_t *dev, uint8_t max_range)
|
2014-12-11 16:13:49 +01:00
|
|
|
{
|
2015-01-20 08:54:40 +01:00
|
|
|
int status;
|
|
|
|
|
|
|
|
/* Acquire exclusive access to the bus. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_acquire(SRF08_DEV_I2C);
|
|
|
|
status = i2c_write_reg(SRF08_DEV_I2C, SRF08_DEV_ADDR, SRF08_RANGE_REG, max_range, 0);
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_release(SRF08_DEV_I2C);
|
2015-01-20 08:54:40 +01:00
|
|
|
|
|
|
|
return status;
|
2014-12-11 16:13:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int srf08_set_max_gain(const srf08_t *dev, uint8_t gain)
|
2014-12-11 16:13:49 +01:00
|
|
|
{
|
2015-01-20 08:54:40 +01:00
|
|
|
int status;
|
|
|
|
|
|
|
|
/* Acquire exclusive access to the bus. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_acquire(SRF08_DEV_I2C);
|
|
|
|
status = i2c_write_reg(SRF08_DEV_I2C, SRF08_DEV_ADDR, SRF08_GAIN_REG, gain, 0);
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_release(SRF08_DEV_I2C);
|
2015-01-20 08:54:40 +01:00
|
|
|
|
|
|
|
return status;
|
2014-12-11 16:13:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-27 08:25:51 +02:00
|
|
|
int srf08_get_distances(const srf08_t *dev, uint16_t *range_array,
|
|
|
|
int num_echos, srf08_mode_t ranging_mode)
|
2014-12-11 16:13:49 +01:00
|
|
|
{
|
|
|
|
int status;
|
|
|
|
int echo_number = 0;
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t range_bytes[sizeof(uint16_t)];
|
|
|
|
uint8_t register_location;
|
2014-12-11 16:13:49 +01:00
|
|
|
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. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_acquire(SRF08_DEV_I2C);
|
2014-12-11 16:13:49 +01:00
|
|
|
/* set ranging mode */
|
2020-02-21 12:05:15 +01:00
|
|
|
status = i2c_write_reg(SRF08_DEV_I2C, SRF08_DEV_ADDR, SRF08_COMMAND_REG,
|
2018-06-27 08:25:51 +02:00
|
|
|
ranging_mode, 0);
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_release(SRF08_DEV_I2C);
|
2014-12-11 16:13:49 +01:00
|
|
|
|
2018-06-27 08:25:51 +02:00
|
|
|
if (status != 0) {
|
2016-09-27 23:38:41 +02:00
|
|
|
DEBUG("Write the ranging command to the i2c-interface is failed\n");
|
2014-12-11 16:13:49 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(max_reg_no_read > SRF08_MAX_REGISTER_NUMBER) {
|
2016-09-27 23:38:41 +02:00
|
|
|
DEBUG("Too many echos requested. Max. is 17\n");
|
2014-12-11 16:13:49 +01:00
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
2015-08-14 10:22:46 +02:00
|
|
|
xtimer_usleep(70000);
|
2014-12-11 16:13:49 +01:00
|
|
|
|
|
|
|
/* 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. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_acquire(SRF08_DEV_I2C);
|
2014-12-11 16:13:49 +01:00
|
|
|
/* read the echo bytes */
|
2020-02-21 12:05:15 +01:00
|
|
|
status = i2c_read_regs(SRF08_DEV_I2C, SRF08_DEV_ADDR, register_location,
|
2018-06-27 08:25:51 +02:00
|
|
|
range_bytes, sizeof(range_bytes), 0);
|
2015-01-20 08:54:40 +01:00
|
|
|
/* Release the bus for other threads. */
|
2020-02-21 12:05:15 +01:00
|
|
|
i2c_release(SRF08_DEV_I2C);
|
2014-12-11 16:13:49 +01:00
|
|
|
|
2018-06-27 08:25:51 +02:00
|
|
|
if (status != 0) {
|
2016-09-27 23:38:41 +02:00
|
|
|
DEBUG("Read the echo bytes from the i2c-interface is failed\n");
|
2014-12-11 16:13:49 +01:00
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
uint16_t distance = (range_bytes[0] << 8) | range_bytes[1];
|
|
|
|
range_array[(register_location - 2) / 2] = distance;
|
|
|
|
echo_number++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return echo_number;
|
|
|
|
}
|