2017-06-21 09:15:49 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Hamburg University of Applied Sciences, Dimitri Nahm
|
|
|
|
*
|
|
|
|
* 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-06-29 16:45:07 +02:00
|
|
|
* @ingroup cpu_atmega_common
|
|
|
|
* @ingroup drivers_periph_i2c
|
2017-06-21 09:15:49 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2019-10-23 21:13:51 +02:00
|
|
|
* @brief Low-level I2C driver implementation for atmega common
|
2017-06-21 09:15:49 +02:00
|
|
|
*
|
2017-06-29 16:45:07 +02:00
|
|
|
* @note This implementation only implements the 7-bit addressing mode.
|
2017-06-21 09:15:49 +02:00
|
|
|
*
|
|
|
|
* @author Dimitri Nahm <dimitri.nahm@haw-hamburg.de>
|
2018-05-30 21:34:22 +02:00
|
|
|
* @author Laurent Navet <laurent.navet@gmail.com>
|
2017-06-21 09:15:49 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2018-05-30 21:34:22 +02:00
|
|
|
#include <errno.h>
|
2017-06-21 09:15:49 +02:00
|
|
|
#include "cpu.h"
|
|
|
|
#include "mutex.h"
|
2017-06-29 16:45:07 +02:00
|
|
|
#include "assert.h"
|
2017-06-21 09:15:49 +02:00
|
|
|
#include "periph/i2c.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
|
2017-06-29 16:45:07 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2018-05-30 21:34:22 +02:00
|
|
|
#include "debug.h"
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2017-06-29 16:45:07 +02:00
|
|
|
#define MT_START 0x08
|
|
|
|
#define MT_START_REPEATED 0x10
|
|
|
|
#define MT_ADDRESS_ACK 0x18
|
|
|
|
#define MT_DATA_ACK 0x28
|
|
|
|
#define MR_ADDRESS_ACK 0x40
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
#define ATMEGA_I2C_FLAG_WRITE 0
|
|
|
|
#define ATMEGA_I2C_FLAG_READ 1
|
|
|
|
|
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
/* static function definitions */
|
|
|
|
static int _start(uint8_t address, uint8_t rw_flag);
|
|
|
|
static int _write(const uint8_t *data, int length);
|
|
|
|
static void _stop(void);
|
2018-05-30 21:34:22 +02:00
|
|
|
static void i2c_poweron(i2c_t dev);
|
|
|
|
|
|
|
|
static mutex_t locks[I2C_NUMOF];
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
/* TODO : 10 bits addresses */
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
void i2c_init(i2c_t dev)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2018-05-30 21:34:22 +02:00
|
|
|
/* check if the line is valid */
|
|
|
|
assert(dev < I2C_NUMOF);
|
|
|
|
|
|
|
|
mutex_init(&locks[dev]);
|
|
|
|
|
2017-06-29 16:45:07 +02:00
|
|
|
/* TWI Bit Rate Register - division factor for the bit rate generator */
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
unsigned long twibrr;
|
2017-06-29 16:45:07 +02:00
|
|
|
/* TWI Prescaler Bits - default 0 */
|
|
|
|
uint8_t twipb = 0;
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* calculate speed configuration */
|
2018-05-30 21:34:22 +02:00
|
|
|
switch (I2C_BUS_SPEED) {
|
2017-06-29 16:45:07 +02:00
|
|
|
|
|
|
|
case I2C_SPEED_LOW:
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if ((CLOCK_CORECLOCK > 20000000UL)
|
|
|
|
|| (CLOCK_CORECLOCK < 1000000UL)) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: bus speed incompatible with core clock\n");
|
|
|
|
return;
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2018-05-30 21:34:22 +02:00
|
|
|
twibrr = ((CLOCK_CORECLOCK / 10000UL) - 16) / (2 * 4); /* CLK Prescaler 4 */
|
2017-06-29 16:45:07 +02:00
|
|
|
twipb = 1;
|
|
|
|
break;
|
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
case I2C_SPEED_NORMAL:
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if ((CLOCK_CORECLOCK > 50000000UL)
|
|
|
|
|| (CLOCK_CORECLOCK < 2000000UL)) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: bus speed incompatible with core clock\n");
|
|
|
|
return;
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
twibrr = ((CLOCK_CORECLOCK / 100000UL) - 16) / 2;
|
2017-06-21 09:15:49 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case I2C_SPEED_FAST:
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if (CLOCK_CORECLOCK < 7500000UL) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: bus speed incompatible with core clock\n");
|
|
|
|
return;
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
twibrr = ((CLOCK_CORECLOCK / 400000UL) - 16) / 2;
|
2017-06-21 09:15:49 +02:00
|
|
|
break;
|
|
|
|
|
2017-06-29 16:45:07 +02:00
|
|
|
case I2C_SPEED_FAST_PLUS:
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if (CLOCK_CORECLOCK < 18000000UL) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: bus speed incompatible with core clock\n");
|
|
|
|
return;
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
twibrr = ((CLOCK_CORECLOCK / 1000000UL) - 16) / 2;
|
2017-06-29 16:45:07 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case I2C_SPEED_HIGH:
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if (CLOCK_CORECLOCK < 62000000UL) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: bus speed incompatible with core clock\n");
|
|
|
|
return;
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
twibrr = ((CLOCK_CORECLOCK / 3400000UL) - 16) / 2;
|
2017-06-29 16:45:07 +02:00
|
|
|
break;
|
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
default:
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] init: invalid bus speed\n");
|
|
|
|
return;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* set pull-up on SCL and SDA */
|
2017-06-29 16:45:07 +02:00
|
|
|
I2C_PORT_REG |= (I2C_PIN_MASK);
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* enable I2C clock */
|
|
|
|
i2c_poweron(dev);
|
|
|
|
|
|
|
|
/* disable device */
|
|
|
|
TWCR &= ~(1 << TWEN);
|
|
|
|
/* configure I2C clock */
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
TWBR = (uint8_t)twibrr; /* Set TWI Bit Rate Register */
|
|
|
|
TWSR &= ~(0x03); /* Reset TWI Prescaler Bits */
|
|
|
|
TWSR |= twipb; /* Set TWI Prescaler Bits */
|
2017-06-21 09:15:49 +02:00
|
|
|
/* enable device */
|
|
|
|
TWCR |= (1 << TWEN);
|
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
int i2c_read_bytes(i2c_t dev, uint16_t addr, void *data, size_t len,
|
|
|
|
uint8_t flags)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2018-05-30 21:34:22 +02:00
|
|
|
assert(dev < I2C_NUMOF);
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
/* Check for unsupported operations */
|
|
|
|
if (flags & I2C_ADDR10) {
|
|
|
|
return -EOPNOTSUPP;
|
2017-11-10 15:32:27 +01:00
|
|
|
}
|
2018-05-30 21:34:22 +02:00
|
|
|
/* Check for wrong arguments given */
|
|
|
|
if (data == NULL || len == 0) {
|
|
|
|
return -EINVAL;
|
2017-11-10 15:32:27 +01:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2017-11-10 15:32:27 +01:00
|
|
|
uint8_t *my_data = data;
|
2018-05-30 21:34:22 +02:00
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
/* send start condition and slave address */
|
2018-05-30 21:34:22 +02:00
|
|
|
if (!(flags & I2C_NOSTART)) {
|
|
|
|
if (_start(addr, ATMEGA_I2C_FLAG_READ) < 0) {
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
for (size_t i = 0; i < len; i++) {
|
2017-06-21 09:15:49 +02:00
|
|
|
/* Send NACK for last received byte */
|
2018-05-30 21:34:22 +02:00
|
|
|
if ((len - i) == 1) {
|
2017-06-21 09:15:49 +02:00
|
|
|
TWCR = (1 << TWEN) | (1 << TWINT);
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
|
|
|
else {
|
2017-06-21 09:15:49 +02:00
|
|
|
TWCR = (1 << TWEA) | (1 << TWEN) | (1 << TWINT);
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] i2c_read_bytes: Wait for byte\n");
|
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
/* Wait for TWINT Flag set. This indicates that DATA has been received.*/
|
2017-06-29 16:45:07 +02:00
|
|
|
while (!(TWCR & (1 << TWINT))) {}
|
2018-05-30 21:34:22 +02:00
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
/* receive data byte */
|
|
|
|
my_data[i] = TWDR;
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] i2c_read_bytes: Byte received\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* end transmission */
|
2018-05-30 21:34:22 +02:00
|
|
|
if (!(flags & I2C_NOSTOP)) {
|
2017-06-21 09:15:49 +02:00
|
|
|
_stop();
|
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
return 0;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data, size_t len,
|
|
|
|
uint8_t flags)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2018-05-30 21:34:22 +02:00
|
|
|
assert(dev < I2C_NUMOF);
|
2017-11-10 15:32:27 +01:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
/* Check for unsupported operations */
|
|
|
|
if (flags & I2C_ADDR10) {
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
|
|
|
/* Check for wrong arguments given */
|
|
|
|
if (data == NULL || len == 0) {
|
|
|
|
return -EINVAL;
|
2017-11-10 15:32:27 +01:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* start transmission and send slave address */
|
2018-05-30 21:34:22 +02:00
|
|
|
if (!(flags & I2C_NOSTART)) {
|
|
|
|
if (_start(addr, ATMEGA_I2C_FLAG_WRITE) < 0) {
|
|
|
|
DEBUG("[i2c] i2c_write_bytes: start failed\n");
|
|
|
|
return -ENXIO;
|
|
|
|
}
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* send out data bytes */
|
2018-05-30 21:34:22 +02:00
|
|
|
if (_write(data, len) < 0) {
|
|
|
|
DEBUG("[i2c] i2c_write_bytes: write failed\n");
|
|
|
|
return -EIO;
|
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* end transmission */
|
2018-05-30 21:34:22 +02:00
|
|
|
if (!(flags & I2C_NOSTOP)) {
|
|
|
|
DEBUG("[i2c] i2c_write_bytes: sending stop\n");
|
|
|
|
_stop();
|
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
return 0;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
int i2c_acquire(i2c_t dev)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2018-05-30 21:34:22 +02:00
|
|
|
assert(dev < I2C_NUMOF);
|
2017-06-21 09:15:49 +02:00
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
mutex_lock(&locks[dev]);
|
|
|
|
return 0;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
2019-08-22 11:45:18 +02:00
|
|
|
void i2c_release(i2c_t dev)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2017-06-29 16:45:07 +02:00
|
|
|
assert(dev < I2C_NUMOF);
|
2018-05-30 21:34:22 +02:00
|
|
|
|
|
|
|
mutex_unlock(&locks[dev]);
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
static void i2c_poweron(i2c_t dev)
|
2017-06-21 09:15:49 +02:00
|
|
|
{
|
2017-06-29 16:45:07 +02:00
|
|
|
assert(dev < I2C_NUMOF);
|
build: fix unused parameter errors
cpu, sam0_common: fix unused parameter in periph/spi
cpu, kinetis_common: fix unused parameter in periph/spi
cpu, cc2538: fix unused param in periph/i2c
cpu, cc2538: fix unused param in periph/spi
cpu, sam3: fix unused param in periph/spi
cpu, stm32_common: fix unused param in periph/pm
cpu, stm32f3: fix unused params in periph/i2c
cpu, nrf5x_common: fix unused param in periph/gpio
cpu, nrf5x_common: fix unused param in periph/spi
cpu, lpc2387: fix unused params in periph/spi
cpu, cc2538: fix unused params in radio/netdev
cpu, cc2650: fix unused params in periph/uart
cpu, lm4f120: fix unused param in periph/spi
cpu, lm4f120: fix unused params in periph/timer
cpu, lm4f120: fix unused params in periph/uart
cpu, stm32_common: fix unused params in periph/dac
cpu, stm32l0: fix unused params in periph/i2c
cpu, msp430fxyz: fix unused params in periph/uart
cpu, mips: fix unused params
cpu, cc430: fix unused-params in periph/timer
cpu, msp430fxyz: fix unused params in periph/spi
drivers, cc2420: fix unused param
cpu, mips32r2_common: fix unused params in periph/timer
cpu, cc2538: fix unused-param in periph/i2c
cpu, mips32r2_common: fix unused-param in periph/timer
cpu, msp430fxyz: fix unused params in periph/timer
cpu, atmega_common: fix unused params in periph/spi
driver, nrfmin: fix unused params
cpu, cc2538_rf: fix unused params
driver, netdev_ieee802514: fix unused param
cpu, mip_pic32m: fix unused params
cpu, lpc2387: fix unused params in periph/pwm
tests/driver_sdcard_spi: fix unused params
cpu, sam3: fix unused param in periph/pwm
tests/driver_dynamixel: fix unused params, and style issues
cpu, cc430: fix unused param in periph/rtc
cpu, atmega_common: fix unused params in periph/i2c
2017-10-31 12:09:11 +01:00
|
|
|
(void) dev;
|
2018-05-30 21:34:22 +02:00
|
|
|
power_twi_enable();
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int _start(uint8_t address, uint8_t rw_flag)
|
|
|
|
{
|
|
|
|
/* Reset I2C Interrupt Flag and transmit START condition */
|
|
|
|
TWCR = (1 << TWINT) | (1 << TWSTA) | (1 << TWEN);
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: START condition transmitted\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Wait for TWINT Flag set. This indicates that the START has been
|
2017-06-29 16:45:07 +02:00
|
|
|
* transmitted, and ACK/NACK has been received.*/
|
|
|
|
while (!(TWCR & (1 << TWINT))) {}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Check value of TWI Status Register. Mask prescaler bits.
|
2017-06-29 16:45:07 +02:00
|
|
|
* If status different from START go to ERROR */
|
|
|
|
if ((TWSR & 0xF8) == MT_START) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: I2C Status is: START\n");
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
|
|
|
else if ((TWSR & 0xF8) == MT_START_REPEATED) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: I2C Status is: START REPEATED\n");
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
|
|
|
else {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: I2C Status Register is different from "
|
|
|
|
"START/START_REPEATED\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
_stop();
|
|
|
|
return -1;
|
|
|
|
}
|
2017-06-29 16:45:07 +02:00
|
|
|
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Load ADDRESS and R/W Flag into TWDR Register.
|
2017-06-29 16:45:07 +02:00
|
|
|
* Clear TWINT bit in TWCR to start transmission of ADDRESS */
|
2017-06-21 09:15:49 +02:00
|
|
|
TWDR = (address << 1) | rw_flag;
|
|
|
|
TWCR = (1 << TWINT) | (1 << TWEN);
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: ADDRESS and FLAG transmitted\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Wait for TWINT Flag set. This indicates that ADDRESS has been transmitted.*/
|
2017-06-29 16:45:07 +02:00
|
|
|
while (!(TWCR & (1 << TWINT))) {}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Check value of TWI Status Register. Mask prescaler bits.
|
2017-06-29 16:45:07 +02:00
|
|
|
* If status different from ADDRESS ACK go to ERROR */
|
|
|
|
if ((TWSR & 0xF8) == MT_ADDRESS_ACK) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: ACK has been received for ADDRESS %02X (write)\n",
|
|
|
|
address);
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
|
|
|
else if ((TWSR & 0xF8) == MR_ADDRESS_ACK) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: ACK has been received for ADDRESS %02X (read)\n",
|
|
|
|
address);
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
else {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] start: NACK has been received for ADDRESS %02X \n",
|
|
|
|
address);
|
2017-06-21 09:15:49 +02:00
|
|
|
_stop();
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
/* TODO : const uint8_t data instead of *data */
|
2017-06-21 09:15:49 +02:00
|
|
|
static int _write(const uint8_t *data, int length)
|
|
|
|
{
|
2017-06-29 16:45:07 +02:00
|
|
|
for (int i = 0; i < length; i++) {
|
2017-06-21 09:15:49 +02:00
|
|
|
/* Load DATA into TWDR Register.
|
2017-06-29 16:45:07 +02:00
|
|
|
* Clear TWINT bit in TWCR to start transmission of data */
|
2017-06-21 09:15:49 +02:00
|
|
|
TWDR = data[i];
|
|
|
|
TWCR = (1 << TWINT) | (1 << TWEN);
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] write: Byte transmitted\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Wait for TWINT Flag set. This indicates that DATA has been transmitted.*/
|
2017-06-29 16:45:07 +02:00
|
|
|
while (!(TWCR & (1 << TWINT))) {}
|
2017-06-21 09:15:49 +02:00
|
|
|
|
|
|
|
/* Check value of TWI Status Register. Mask prescaler bits. If status
|
2017-06-29 16:45:07 +02:00
|
|
|
* different from MT_DATA_ACK, return number of transmitted bytes */
|
2017-06-21 09:15:49 +02:00
|
|
|
if ((TWSR & 0xF8) != MT_DATA_ACK) {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] write: NACK has been received\n");
|
|
|
|
return -1;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
2017-06-29 16:45:07 +02:00
|
|
|
else {
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] write: ACK has been received\n");
|
2017-06-29 16:45:07 +02:00
|
|
|
}
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
return 0;
|
2017-06-21 09:15:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void _stop(void)
|
|
|
|
{
|
|
|
|
/* Reset I2C Interrupt Flag and transmit STOP condition */
|
|
|
|
TWCR = (1 << TWINT) | (1 << TWSTO) | (1 << TWEN);
|
2017-06-29 16:45:07 +02:00
|
|
|
/* Wait for STOP Flag reset. This indicates that STOP has been transmitted.*/
|
|
|
|
while (TWCR & (1 << TWSTO)) {}
|
2018-05-30 21:34:22 +02:00
|
|
|
DEBUG("[i2c] stop: STOP condition transmitted\n");
|
2017-06-21 09:15:49 +02:00
|
|
|
TWCR = 0;
|
|
|
|
}
|