mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 09:32:43 +01:00
152 lines
4.0 KiB
C
152 lines
4.0 KiB
C
/*
|
|
* Copyright (C) 2013 Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
|
|
* 2015 Freie Universität Berlin
|
|
*
|
|
* 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 drivers_srf02
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Driver for the SRF02 ultrasonic range sensor
|
|
*
|
|
* @author Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
* @author Kevin Weiss <kevin.weiss@haw-hamburg.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
|
|
#include "xtimer.h"
|
|
#include "srf02.h"
|
|
#include "periph/i2c.h"
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#include "debug.h"
|
|
|
|
/**
|
|
* @brief SRF02 register addresses
|
|
* @{
|
|
*/
|
|
#define REG_CMD (0x00)
|
|
#define REG_HIGH (0x02)
|
|
#define REG_LOW (0x03)
|
|
#define REG_AUTO_HIGH (0x04)
|
|
#define REG_AUTO_LOW (0x05)
|
|
/** @} */
|
|
|
|
/**
|
|
* @brief Some additional SRF02 commands
|
|
* @{
|
|
*/
|
|
#define CMD_ADDR_SEQ1 (0xa0)
|
|
#define CMD_ADDR_SEQ2 (0xaa)
|
|
#define CMD_ADDR_SEQ3 (0xa5)
|
|
/** @} */
|
|
|
|
|
|
int srf02_init(srf02_t *dev, i2c_t i2c, uint8_t addr)
|
|
{
|
|
dev->i2c = i2c;
|
|
dev->addr = (addr >> 1); /* internally we right align the 7-bit addr */
|
|
uint8_t rev = 0;
|
|
|
|
/* Acquire exclusive access to the bus. */
|
|
i2c_acquire(dev->i2c);
|
|
|
|
/* try to read the software revision (read the CMD reg) from the device */
|
|
i2c_read_reg(i2c, dev->addr, REG_CMD, &rev, 0);
|
|
if (rev == 0 || rev == 255) {
|
|
i2c_release(dev->i2c);
|
|
DEBUG("[srf02] error reading the devices software revision\n");
|
|
return -1;
|
|
} else {
|
|
DEBUG("[srf02] software revision: 0x%02x\n", rev);
|
|
}
|
|
/* Release the bus for other threads. */
|
|
i2c_release(dev->i2c);
|
|
|
|
DEBUG("[srf02] initialization successful\n");
|
|
return 0;
|
|
}
|
|
|
|
int srf02_trigger(const srf02_t *dev, srf02_mode_t mode)
|
|
{
|
|
int status;
|
|
/* trigger a new measurement by writing the mode to the CMD register */
|
|
DEBUG("[srf02] trigger new reading\n");
|
|
i2c_acquire(dev->i2c);
|
|
status = i2c_write_reg(dev->i2c, dev->addr, REG_CMD, mode, 0);
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|
|
|
|
uint16_t srf02_read(const srf02_t *dev)
|
|
{
|
|
uint8_t res[2] = {0xFF, 0xFF};
|
|
|
|
/* read the results */
|
|
i2c_acquire(dev->i2c);
|
|
i2c_read_regs(dev->i2c, dev->addr, REG_HIGH, res, 2, 0);
|
|
i2c_release(dev->i2c);
|
|
DEBUG("[srf02] result - high: 0x%02x low: 0x%02x\n", res[0], res[1]);
|
|
|
|
/* compile result - TODO: fix for different host byte order other than LE */
|
|
return ((((uint16_t)res[0]) << 8) | (res[1] & 0xff));
|
|
}
|
|
|
|
uint16_t srf02_get_distance(const srf02_t *dev, srf02_mode_t mode)
|
|
{
|
|
/* trigger a new reading */
|
|
srf02_trigger(dev, mode);
|
|
/* give the sensor the required time for sampling */
|
|
xtimer_usleep(SRF02_RANGE_DELAY);
|
|
/* get the results */
|
|
return srf02_read(dev);
|
|
}
|
|
|
|
int srf02_set_addr(srf02_t *dev, uint8_t new_addr)
|
|
{
|
|
int status;
|
|
/* get access to the bus */
|
|
i2c_acquire(dev->i2c);
|
|
|
|
DEBUG("[srf02] reprogramming device address to 0x%02x\n", (int)new_addr);
|
|
|
|
/* write the new address, for this we need to follow a certain sequence */
|
|
status = i2c_write_reg(dev->i2c, dev->addr, REG_CMD, CMD_ADDR_SEQ1, 0);
|
|
if (status != 0) {
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|
|
status = i2c_write_reg(dev->i2c, dev->addr, REG_CMD, CMD_ADDR_SEQ2, 0);
|
|
if (status != 0) {
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|
|
status = i2c_write_reg(dev->i2c, dev->addr, REG_CMD, CMD_ADDR_SEQ3, 0);
|
|
if (status != 0) {
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|
|
status = i2c_write_reg(dev->i2c, dev->addr, REG_CMD, new_addr, 0);
|
|
if (status != 0) {
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|
|
|
|
dev->addr = (new_addr >> 1);
|
|
|
|
/* release the bus */
|
|
i2c_release(dev->i2c);
|
|
return status;
|
|
}
|