1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

ds1307: initial import of a driver for the DS1307 RTC

This commit is contained in:
Martine Lenders 2017-07-04 13:28:02 +02:00
parent ef9007b56c
commit d07bfb4c9f
No known key found for this signature in database
GPG Key ID: 8E97A9FE55F25D62
7 changed files with 512 additions and 0 deletions

View File

@ -84,6 +84,10 @@ ifneq (,$(filter dht,$(USEMODULE)))
FEATURES_REQUIRED += periph_gpio
endif
ifneq (,$(filter ds1307,$(USEMODULE)))
FEATURES_REQUIRED += periph_i2c
endif
ifneq (,$(filter dsp0401,$(USEMODULE)))
USEMODULE += xtimer
FEATURES_REQUIRED += periph_gpio

View File

@ -10,6 +10,9 @@ endif
ifneq (,$(filter cc110x,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/cc110x/include
endif
ifneq (,$(filter ds1307,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/ds1307/include
endif
ifneq (,$(filter kw2xrf,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/kw2xrf/include
endif

1
drivers/ds1307/Makefile Normal file
View File

@ -0,0 +1 @@
include $(RIOTBASE)/Makefile.base

213
drivers/ds1307/ds1307.c Normal file
View File

@ -0,0 +1,213 @@
/*
* Copyright (C) 2017 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.
*/
/**
* @{
*
* @file
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
#include <string.h>
#include "bcd.h"
#include "ds1307.h"
#include "ds1307_internal.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
static int _nvram_read(struct nvram *dev, uint8_t *dst, uint32_t src,
size_t size);
static int _nvram_write(struct nvram *dev, const uint8_t *src, uint32_t dst,
size_t size);
static uint8_t _convert_12_to_24(uint8_t hour)
{
if (hour & DS1307_REG_HOUR_12H) {
uint8_t tmp = bcd_to_byte(hour & DS1307_REG_HOUR_12H_MASK);
if (hour & DS1307_REG_HOUR_PM) {
if (tmp < 12) {
tmp += 12;
}
}
else {
if (tmp == 12) {
tmp = 0;
}
}
hour = (bcd_from_byte(tmp) & DS1307_REG_HOUR_24H_MASK);
}
return hour;
}
int ds1307_init(ds1307_t *dev, const ds1307_params_t *params)
{
int res;
uint8_t hour;
dev->i2c = params->i2c;
i2c_acquire(dev->i2c);
res = i2c_init_master(dev->i2c, params->clk);
if (res < 0) {
i2c_release(dev->i2c);
DEBUG("ds1307: Error initializing I2C: %i\n", res);
return -1;
}
/* normalize hour format */
res = i2c_read_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_HOUR, &hour);
if (res < 0) {
i2c_release(dev->i2c);
DEBUG("ds1307: Error reading HOUR register on init: %i\n", res);
return -1;
}
res = i2c_write_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_HOUR,
_convert_12_to_24(hour));
i2c_release(dev->i2c);
if (res < 0) {
DEBUG("ds1307: Error writing HOUR register on init: %i\n", res);
return -1;
}
dev->nvram.read = _nvram_read;
dev->nvram.write = _nvram_write;
dev->nvram.size = (DS1307_REG_RAM_LAST - DS1307_REG_RAM_FIRST) + 1;
dev->nvram.extra = dev;
return 0;
}
int ds1307_set_time(const ds1307_t *dev, const struct tm *time)
{
uint8_t regs[DS1307_REG_YEAR - DS1307_REG_SEC + 1];
int res;
regs[DS1307_REG_SEC] = (bcd_from_byte(time->tm_sec) & DS1307_REG_SEC_MASK);
regs[DS1307_REG_MIN] = (bcd_from_byte(time->tm_min) & DS1307_REG_MIN_MASK);
regs[DS1307_REG_HOUR] = (bcd_from_byte(time->tm_hour) &
DS1307_REG_HOUR_24H_MASK);
regs[DS1307_REG_DOW] = (bcd_from_byte(time->tm_wday + DS1307_DOW_OFFSET) &
DS1307_REG_DOW_MASK);
regs[DS1307_REG_DOM] = (bcd_from_byte(time->tm_mday) & DS1307_REG_DOM_MASK);
regs[DS1307_REG_MON] = (bcd_from_byte(time->tm_mon + DS1307_MON_OFFSET) &
DS1307_REG_MON_MASK);
regs[DS1307_REG_YEAR] = bcd_from_byte(time->tm_year + DS1307_YEAR_OFFSET);
i2c_acquire(dev->i2c);
res = i2c_write_regs(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SEC, regs,
sizeof(regs));
DEBUG("ds1307: wrote bytes %02x %02x %02x %02x %02x %02x %02x to device (result: %i)\n",
regs[DS1307_REG_SEC], regs[DS1307_REG_MIN], regs[DS1307_REG_HOUR],
regs[DS1307_REG_DOW], regs[DS1307_REG_DOM], regs[DS1307_REG_MON],
regs[DS1307_REG_YEAR], res);
i2c_release(dev->i2c);
return (res < 0) ? -1 : 0;
}
int ds1307_get_time(const ds1307_t *dev, struct tm *time)
{
uint8_t regs[DS1307_REG_YEAR - DS1307_REG_SEC + 1];
int res;
i2c_acquire(dev->i2c);
res = i2c_read_regs(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SEC, regs,
sizeof(regs));
DEBUG("ds1307: read bytes %02x %02x %02x %02x %02x %02x %02x from device (result: %i)\n",
regs[DS1307_REG_SEC], regs[DS1307_REG_MIN], regs[DS1307_REG_HOUR],
regs[DS1307_REG_DOW], regs[DS1307_REG_DOM], regs[DS1307_REG_MON],
regs[DS1307_REG_YEAR], res);
i2c_release(dev->i2c);
if (res < 0) {
return -1;
}
time->tm_sec = bcd_to_byte(regs[DS1307_REG_SEC] & DS1307_REG_SEC_MASK);
time->tm_min = bcd_to_byte(regs[DS1307_REG_MIN] & DS1307_REG_MIN_MASK);
time->tm_hour = bcd_to_byte(regs[DS1307_REG_HOUR] &
DS1307_REG_HOUR_24H_MASK);
time->tm_wday = (bcd_to_byte(regs[DS1307_REG_DOW] & DS1307_REG_DOW_MASK) -
DS1307_DOW_OFFSET);
time->tm_mday = bcd_to_byte(regs[DS1307_REG_DOM] & DS1307_REG_DOM_MASK);
time->tm_mon = bcd_to_byte(regs[DS1307_REG_MON] & DS1307_REG_MON_MASK) -
DS1307_MON_OFFSET;
time->tm_year = (bcd_to_byte(regs[DS1307_REG_YEAR]) - DS1307_YEAR_OFFSET);
return 0;
}
int ds1307_halt(const ds1307_t *dev)
{
int res;
uint8_t sec;
i2c_acquire(dev->i2c);
res = i2c_read_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SEC, &sec);
if (res < 0) {
i2c_release(dev->i2c);
DEBUG("ds1307: Error reading SEC register on halt: %i\n", res);
return -1;
}
sec |= DS1307_REG_SEC_CH;
res = i2c_write_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SEC, sec);
i2c_release(dev->i2c);
return (res < 0) ? -1 : 0;
}
int ds1307_set_sqw_mode(const ds1307_t *dev, ds1307_sqw_mode_t mode)
{
int res;
i2c_acquire(dev->i2c);
res = i2c_write_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SQW_CTL,
(uint8_t)mode);
i2c_release(dev->i2c);
return res;
}
int ds1307_get_sqw_mode(const ds1307_t *dev)
{
uint8_t mode;
int res;
i2c_acquire(dev->i2c);
res = i2c_read_reg(dev->i2c, DS1307_I2C_ADDRESS, DS1307_REG_SQW_CTL, &mode);
i2c_release(dev->i2c);
return (res < 0) ? res : (int)mode;
}
static int _nvram_read(struct nvram *nvram, uint8_t *dst, uint32_t src,
size_t size)
{
const ds1307_t *dev = nvram->extra;
int res;
if ((src + size) > nvram->size) {
return -3;
}
i2c_acquire(dev->i2c);
res = i2c_read_regs(dev->i2c, DS1307_I2C_ADDRESS,
DS1307_REG_RAM_FIRST + src, dst, size);
i2c_release(dev->i2c);
return res;
}
static int _nvram_write(struct nvram *nvram, const uint8_t *src, uint32_t dst,
size_t size)
{
const ds1307_t *dev = nvram->extra;
int res;
if ((dst + size) > nvram->size) {
return -3;
}
i2c_acquire(dev->i2c);
res = i2c_write_regs(dev->i2c, DS1307_I2C_ADDRESS,
DS1307_REG_RAM_FIRST + dst, src, size);
i2c_release(dev->i2c);
return res;
}
/** @} */

View File

@ -0,0 +1,73 @@
/*
* Copyright (C) 2017 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_ds1307
* @brief
* @{
*
* @file
* @brief Register definitions for DS1307 RTC
*
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
#ifndef DS1307_INTERNAL_H
#define DS1307_INTERNAL_H
#ifdef __cplusplus
extern "C" {
#endif
/**
* @name Registers
* @{
*/
#define DS1307_REG_SEC (0x00) /**< seconds */
#define DS1307_REG_MIN (0x01) /**< minutes */
#define DS1307_REG_HOUR (0x02) /**< hours */
#define DS1307_REG_DOW (0x03) /**< day of week (1-7, 1 == sunday) */
#define DS1307_REG_DOM (0x04) /**< day of month */
#define DS1307_REG_MON (0x05) /**< month */
#define DS1307_REG_YEAR (0x06) /**< year */
#define DS1307_REG_SQW_CTL (0x07) /**< SQW control */
#define DS1307_REG_RAM_FIRST (0x08) /**< NVRAM start */
#define DS1307_REG_RAM_LAST (0x3F) /**< NVRAM end */
/** @} */
/**
* @name Register bits
* @{
*/
#define DS1307_REG_SEC_CH (0x80) /**< clock halt bit */
#define DS1307_REG_SEC_MASK (0x7f) /**< seconds mask */
#define DS1307_REG_MIN_MASK (0x7f) /**< minutes mask */
#define DS1307_REG_HOUR_12H (0x40) /**< 12-hour (0) / 24-hour (1) mode */
#define DS1307_REG_HOUR_PM (0x20) /**< AM (0) / PM (1) in 12-hour mode */
#define DS1307_REG_HOUR_12H_MASK (0x2f) /**< hour (12-hour mode) */
#define DS1307_REG_HOUR_24H_MASK (0x3f) /**< hour (24-hour mode) */
#define DS1307_REG_DOW_MASK (0x07) /**< day of week mask */
#define DS1307_REG_DOM_MASK (0x3f) /**< day of month mask */
#define DS1307_REG_MON_MASK (0x1f) /**< month mask */
/** @} */
/**
* @name Custom offsets (to DS1307 registers to struct tm)
* @{
*/
#define DS1307_DOW_OFFSET (1) /**< offset in days from sunday */
#define DS1307_MON_OFFSET (1) /**< offset in month from January */
#define DS1307_YEAR_OFFSET (-100) /**< offset in years from 1900 */
/** @} */
#ifdef __cplusplus
}
#endif
#endif /* DS1307_INTERNAL_H */
/** @} */

View File

@ -0,0 +1,61 @@
/*
* Copyright (C) 2017 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_ds1307
* @{
*
* @file
* @brief Default configuration for DS1307 devices
*
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
#ifndef DS1307_PARAMS_H
#define DS1307_PARAMS_H
#include "board.h"
#include "ds1307.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @name Default configuration parameters for the DS1307 driver
* @{
*/
#ifndef DS1307_PARAM_I2C
#define DS1307_PARAM_I2C (I2C_DEV(0))
#endif
#ifndef DS1307_PARAM_I2C_CLK
#define DS1307_PARAM_I2C_CLK (DS1307_I2C_MAX_CLK)
#define DS1307_PARAMS_DEFAULT { .i2c = DS1307_PARAM_I2C, \
.clk = DS1307_PARAM_I2C_CLK }
#endif
/** @} */
/**
* @brief DS1307 configuration
*/
static const ds1307_params_t ds1307_params[] =
{
#ifdef DS1307_PARAMS_BOARD
DS1307_PARAMS_BOARD,
#else
DS1307_PARAMS_DEFAULT,
#endif
};
#ifdef __cplusplus
}
#endif
#endif /* DS1307_PARAMS_H */
/** @} */

157
drivers/include/ds1307.h Normal file
View File

@ -0,0 +1,157 @@
/*
* Copyright (C) 2017 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.
*/
/**
* @defgroup drivers_ds1307 DS1307 RTC
* @ingroup drivers_sensors
* @brief Device drive interface for the DS1307 real-time clock
* @{
*
* @file
* @brief DS1307 definitions
*
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
#ifndef DS1307_H
#define DS1307_H
#include <stdbool.h>
#include <time.h>
#include "nvram.h"
#include "periph/i2c.h"
#include "periph/gpio.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief I2C address of DS1307 RTC
*/
#ifndef DS1307_I2C_ADDRESS
#define DS1307_I2C_ADDRESS (0x68)
#endif
/**
* @brief Maximum I2C bus speed to use with the device
*/
#ifndef DS1307_I2C_MAX_CLK
#define DS1307_I2C_MAX_CLK (I2C_SPEED_FAST)
#endif
/**
* @brief Maximum size in byte of on-chip NVRAM
*/
#define DS1307_NVRAM_MAX_SIZE (56U)
/**
* @brief Device descriptor for DS1307 devices
*/
typedef struct {
i2c_t i2c; /**< I2C bus the device is connected to */
nvram_t nvram; /**< on-chip NVRAM (see nvram.h) */
} ds1307_t;
/**
* @brief Set of configuration parameters for DS1307 devices
*/
typedef struct {
i2c_t i2c; /**< I2C bus the device is connected to */
i2c_speed_t clk; /**< clock speed to use on the I2C bus */
} ds1307_params_t;
/**
* @brief Modes for the DS1307 square wave / output driver
*/
typedef enum {
DS1307_SQW_MODE_0 = 0x00, /**< OUT: 0 */
DS1307_SQW_MODE_1000HZ = 0x10, /**< SQW: 1kHz */
DS1307_SQW_MODE_4096HZ = 0x11, /**< SQW: 4.096 kHz */
DS1307_SQW_MODE_8192HZ = 0x12, /**< SQW: 8.192 kHz */
DS1307_SQW_MODE_32768HZ = 0x13, /**< SQW: 32.768 kHz */
DS1307_SQW_MODE_1 = 0x80, /**< OUT: 1 */
} ds1307_sqw_mode_t;
/**
* @brief Initialize the given DS1307 device
*
* @param[out] dev device descriptor of the targeted device
* @param[in] params device configuration (i2c bus, address and bus clock)
*
* @return 0 on success
* @return < 0 if unable to speak to the device
*/
int ds1307_init(ds1307_t *dev, const ds1307_params_t *params);
/**
* @brief Set RTC to a given time.
*
* @param[in] dev device descriptor of the targeted device
* @param[in] time pointer to the struct holding the time to set.
*
* @return 0 on success
* @return < 0 if unable to speak to the device
*/
int ds1307_set_time(const ds1307_t *dev, const struct tm *time);
/**
* @brief Get current RTC time.
*
* @param[in] dev device descriptor of the targeted device
* @param[out] time pointer to the struct to write the time to.
*
* @return 0 on success
* @return < 0 if unable to speak to the device
*/
int ds1307_get_time(const ds1307_t *dev, struct tm *time);
/**
* @brief Halt clock
*
* @note Can be reversed using @ref ds1307_set_time()
*
* @param[in] dev device descriptor of the targeted device
*
* @return 0 on success
* @return < 0 if unable to speak to the device
*/
int ds1307_halt(const ds1307_t *dev);
/**
* @brief Set mode of square wave / output driver
*
* @note To get the actual output of the driver, attach the pin via GPIO
*
* @param[in] dev device descriptor of the targeted device
* @param[in] mode mode for the square wave / output driver
*
* @return 0 on success
* @return < 0 if unable to speak to the device
*/
int ds1307_set_sqw_mode(const ds1307_t *dev, ds1307_sqw_mode_t mode);
/**
* @brief Get current mode of square wave / output driver
*
* @note To get the actual output of the driver, attach the pin via GPIO
*
* @param[in] dev device descriptor of the targeted device
*
* @return current mode of the square wave / output driver
* (see ds1307_sqw_mode_t)
* @return < 0 if unable to speak to the device
*/
int ds1307_get_sqw_mode(const ds1307_t *dev);
#ifdef __cplusplus
}
#endif
#endif /* DS1307_H */
/** @} */