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

947 lines
28 KiB
C
Raw Normal View History

2013-08-07 16:54:10 +02:00
/*
* i2c.c - implementation of the I2C interface for the LPC2387 chip.
* 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.
2013-08-07 16:54:10 +02:00
*/
/**
* @file
* @internal
2013-08-27 15:52:20 +02:00
* @brief The I2C interface drivers for the LPC2387 chip.
* The driver is full abstracted supporting all the i2c-interfaces
* of the LPC2387 chip. The user need only to give the requested
* i2c-interface and the transmission baudrate.
* The user can optionally give a master interrupt handler. If the
* user does not declare a handler, an appropriate interrupt is
* automatically registered for the specific i2c interface.
2013-08-07 16:54:10 +02:00
*
2013-08-27 15:52:20 +02:00
* @author Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
* @author Marco Ziegert <ziegert@inf.fu-berlin.de>
* @author Benjamin Aschenbrenner
* @version $Revision: 3858 $
2013-08-07 16:54:10 +02:00
*
2013-08-27 15:52:20 +02:00
* @note $Id: i2c.c 3858 2013-09-02 18:11:17 kasmi $
2013-08-07 16:54:10 +02:00
*/
#include "lpc23xx.h"
#include "lpc2387.h"
#include "i2c.h"
#include "VIC.h"
#include "irq.h"
#include <stdio.h>
#include <stdbool.h>
#include <stdint.h>
#include <string.h>
volatile uint32_t i2c_master_state = I2C_IDLE;
volatile uint32_t i2c_slave_state = I2C_IDLE;
volatile uint32_t i2c_cmd;
volatile uint32_t i2c_mode;
2013-08-27 15:52:20 +02:00
volatile uint8_t i2c_master_buffer[I2C_BUFSIZE];
2013-08-07 16:54:10 +02:00
volatile uint32_t i2c_read_length;
volatile uint32_t i2c_write_length;
volatile uint32_t rd_index = 0;
volatile uint32_t wr_index = 0;
2013-08-27 15:52:20 +02:00
static void i2c_interface0_master_handler(void) __attribute__((interrupt(
"IRQ")));
static void i2c_interface1_master_handler(void) __attribute__((interrupt(
"IRQ")));
static void i2c_interface2_master_handler(void) __attribute__((interrupt(
"IRQ")));
2013-08-07 16:54:10 +02:00
bool i2c_initialize(uint8_t i2c_interface, uint32_t i2c_mode,
2013-08-27 15:52:20 +02:00
uint8_t slave_addr, uint32_t baud_rate, void *handler)
{
//puts("i2c_initialize begin...\n");
printf("i2cInterface = %d\n", i2c_interface);
i2c_clear_buffer((uint8_t *) i2c_master_buffer,
I2C_BUFSIZE * sizeof(uint8_t));
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//activate power for I2C2
i2c_active_power(i2c_interface);
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//select I2C2 functionality for pins x.xx (SDAx) and x.xx (SCLx)
i2c_pin_select(i2c_interface);
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
// clear I2CCON register flags
i2c_clear_control_register(i2c_interface);
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//set baud rate
i2c_set_baud_rate(i2c_interface, baud_rate);
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//set slave mode
if (i2c_mode == I2CSLAVE) {
i2c_set_slave_mode(i2c_interface, slave_addr);
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
/* Install interrupt handler */
if (!i2c_irq_handler_register(i2c_interface, handler)) {
return false;
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
i2c_initial_master_transmitter_mode(i2c_interface);
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//puts("...i2c_initialize ended\n");
return (true);
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
bool i2c_transaction(uint8_t i2c_interface)
{
//puts("i2cTransaction begin...\n");
i2c_master_state = I2C_IDLE;
rd_index = 0;
wr_index = 0;
if (i2c_start(i2c_interface) != true) {
i2c_stop(i2c_interface);
puts("i2cTransaction return false...\n");
return (false);
}
// puts("entering engine main loop\n");
while (1) {
if (i2c_master_state == DATA_NACK) {
i2c_stop(i2c_interface);
break;
}
}
// puts("...i2cTransaction ended\n");
return (true);
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
bool i2c_start(uint8_t i2c_interface)
{
// puts("i2c_start begin...\n");
uint32_t timeout = 0;
bool retVal = false;
/*--- Issue a start condition ---*/
switch (i2c_interface) {
case I2C0:
I20CONSET = I2CONSET_STA; /* Set Start flag */
break;
case I2C1_0:
case I2C1_1:
I21CONSET = I2CONSET_STA; /* Set Start flag */
break;
case I2C2:
I22CONSET = I2CONSET_STA; /* Set Start flag */
}
/*--- Wait until START transmitted ---*/
while (1) {
if (i2c_master_state == I2C_STARTED) {
retVal = true;
break;
}
if (timeout >= MAX_TIMEOUT) {
puts("timeout");
retVal = false;
break;
}
timeout++;
}
//puts("...i2c_start ended\n");
return (retVal);
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
bool i2c_stop(uint8_t i2c_interface)
{
//puts("i2c_stop begin...\n");
switch (i2c_interface) {
case I2C0:
I20CONSET = I2CONSET_STO; /* Set Stop flag */
I20CONCLR = I2CONCLR_SIC; /* Clear SI flag */
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
/*--- Wait for STOP detected ---*/
while (I20CONSET & I2CONSET_STO)
;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
break;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
case I2C1_0:
case I2C1_1:
I21CONSET = I2CONSET_STO; /* Set Stop flag */
I21CONCLR = I2CONCLR_SIC; /* Clear SI flag */
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
/*--- Wait for STOP detected ---*/
while (I21CONSET & I2CONSET_STO)
;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
break;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
case I2C2:
I22CONSET = I2CONSET_STO; /* Set Stop flag */
I22CONCLR = I2CONCLR_SIC; /* Clear SI flag */
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
/*--- Wait for STOP detected ---*/
while (I22CONSET & I2CONSET_STO)
;
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
// puts("...i2c_stop ended\n");
return true;
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
void i2c_active_power(uint8_t i2c_interface)
{
switch (i2c_interface) {
case I2C0:
PCONP |= BIT7;
break;
case I2C1_0:
case I2C1_1:
PCONP |= BIT19;
break;
case I2C2:
PCONP |= BIT26;
}
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
//select I2C2 functionality for pins x.xx (SDAx) and x.xx (SCLx)
void i2c_pin_select(uint8_t i2cInterface)
{
switch (i2cInterface) {
case I2C0: // P0.27 SDA0, P0.28 SCL0
PINSEL1 |= BIT22 | BIT24;
PINSEL1 &= ~(BIT23 | BIT25);
break;
case I2C1_0: // P0.0 SDA1, P0.1 SCL1
PINSEL0 |= BIT0 | BIT1 | BIT2 | BIT3;
break;
case I2C1_1: // P0.19 SDA1, P0.20 SCL1
PINSEL1 |= BIT6 | BIT7 | BIT8 | BIT9;
break;
case I2C2: // P0.10 SDA2, P0.11 SCL2
PINSEL0 |= BIT21 | BIT23;
PINSEL0 &= ~(BIT20 | BIT22);
}
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
/*--- clearing of bits in the I2CON register ---*/
void i2c_clear_control_register(uint8_t i2c_interface)
{
switch (i2c_interface) {
case I2C0:
I20CONCLR = I2CONCLR_AAC | I2CONCLR_SIC | I2CONCLR_STAC |
I2CONCLR_I2ENC;
break;
case I2C1_0:
case I2C1_1:
I21CONCLR = I2CONCLR_AAC | I2CONCLR_SIC | I2CONCLR_STAC |
I2CONCLR_I2ENC;
break;
case I2C2:
I22CONCLR = I2CONCLR_AAC | I2CONCLR_SIC | I2CONCLR_STAC |
I2CONCLR_I2ENC;
}
}
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
void i2c_set_baud_rate(uint8_t i2c_interface, uint32_t baud_rate)
{
uint32_t pclksel = 0;
uint32_t prescale = 0;
2016-01-05 13:29:52 +01:00
lpc2387_pclk_scale(CLOCK_CORECLOCK, baud_rate, &pclksel, &prescale);
2013-08-27 15:52:20 +02:00
switch (i2c_interface) {
case I2C0:
PCLKSEL0 &= ~(BIT14 | BIT15); //clear Bits
PCLKSEL0 |= pclksel << 14; //set bits
I20SCLL = prescale / 2;
I20SCLH = prescale / 2;
break;
case I2C1_0:
case I2C1_1:
PCLKSEL1 &= ~(BIT6 | BIT7);
PCLKSEL1 |= pclksel << 6;
I21SCLL = prescale / 2;
I21SCLH = prescale / 2;
break;
case I2C2:
PCLKSEL1 &= ~(BIT20 | BIT21);
PCLKSEL1 |= pclksel << 20;
I22SCLL = prescale / 2;
I22SCLH = prescale / 2;
}
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
bool i2c_irq_handler_register(uint8_t i2c_interface, void *handler)
{
bool successful = false;
switch (i2c_interface) {
case I2C0:
if (handler == NULL) {
successful = install_irq(I2C0_INT,
(void *) i2c_interface0_master_handler,
HIGHEST_PRIORITY);
}
else {
successful = install_irq(I2C0_INT, (void *) handler,
HIGHEST_PRIORITY);
}
break;
case I2C1_0:
case I2C1_1:
if (handler == NULL) {
successful = install_irq(I2C1_INT,
(void *) i2c_interface1_master_handler,
HIGHEST_PRIORITY);
}
else {
successful = install_irq(I2C1_INT, (void *) handler,
HIGHEST_PRIORITY);
}
break;
case I2C2:
if (handler == NULL) {
successful = install_irq(I2C2_INT,
(void *) i2c_interface2_master_handler,
HIGHEST_PRIORITY);
}
else {
successful = install_irq(I2C2_INT, (void *) handler,
HIGHEST_PRIORITY);
}
}
return successful;
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
void i2c_initial_master_transmitter_mode(uint8_t i2c_interface)
{
switch (i2c_interface) {
case I2C0:
I20CONSET = I2CONSET_I2EN;
break;
case I2C1_0:
case I2C1_1:
I21CONSET = I2CONSET_I2EN;
break;
case I2C2:
I22CONSET = I2CONSET_I2EN;
}
2013-08-07 16:54:10 +02:00
}
bool i2c_read(uint8_t i2c_interface, uint8_t slave_addr, uint8_t reg_addr,
2013-08-27 15:52:20 +02:00
uint8_t *rx_buff, uint8_t rx_buff_length)
{
i2c_clear_buffer((uint8_t *) i2c_master_buffer,
I2C_BUFSIZE * sizeof(uint8_t));
i2c_write_length = 1;
i2c_read_length = rx_buff_length;
bool successful;
2013-08-27 15:52:20 +02:00
uint8_t readIndex = 3;
i2c_master_buffer[0] = (slave_addr << 1) & WRITE_ENABLE_BIT_MASK;
i2c_master_buffer[1] = reg_addr;
i2c_master_buffer[2] = ((slave_addr << 1) & WRITE_ENABLE_BIT_MASK)
| READ_ENABLE_BIT_MASK;
successful = i2c_transaction(i2c_interface);
if (successful && (rx_buff != NULL) &&
(rx_buff_length < (I2C_BUFSIZE - readIndex))) {
memcpy(rx_buff, (const uint8_t *)(i2c_master_buffer + readIndex),
sizeof(uint8_t) * rx_buff_length);
return true;
}
else {
return false;
}
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
void i2c_clear_buffer(void *ptr, uint32_t size)
{
memset(ptr, 0, size);
2013-08-07 16:54:10 +02:00
}
bool i2c_write(uint8_t i2c_interface, uint8_t slave_addr, uint8_t reg_addr,
2013-08-27 15:52:20 +02:00
uint8_t *tx_buff, uint8_t tx_buff_length)
{
//puts("[i2c.c/i2cWrite]: entered\n");
i2c_clear_buffer((uint8_t *) i2c_master_buffer,
I2C_BUFSIZE * sizeof(uint8_t));
i2c_write_length = tx_buff_length + 1;
i2c_master_buffer[0] = (slave_addr << 1) & WRITE_ENABLE_BIT_MASK;
i2c_master_buffer[1] = reg_addr;
if ((tx_buff != NULL) && tx_buff_length < (I2C_BUFSIZE - 2)) {
int32_t j = 0;
for (int32_t i = 2; i < tx_buff_length + 2; i++) {
2013-08-27 15:52:20 +02:00
i2c_master_buffer[i] = tx_buff[j];
j++;
//printf("I2CMasterBuffer[%d] = %d\n", i, I2CMasterBuffer[i]);
}
return i2c_transaction(i2c_interface);
}
else {
puts("[i2c.c/i2cWrite]: Invalid buffer or invalid write buffer size\n");
return false;
}
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
//burst mode, the first element in the array
2013-08-07 16:54:10 +02:00
bool i2c_trans_receive(uint8_t i2c_interface, uint8_t slave_addr,
2013-08-27 15:52:20 +02:00
uint8_t *tx_buff, uint8_t tx_buff_length,
uint8_t *rx_buff, uint8_t rx_buff_length)
{
puts("[i2c.c/i2cTransReceive]: entered\n");
i2c_clear_buffer((uint8_t *) i2c_master_buffer,
I2C_BUFSIZE * sizeof(uint8_t));
i2c_write_length = tx_buff_length;
i2c_read_length = rx_buff_length;
if (tx_buff != NULL && (tx_buff_length > 0)) {
int32_t read_index = 0;
2013-08-27 15:52:20 +02:00
i2c_master_buffer[0] = (slave_addr << 1) & WRITE_ENABLE_BIT_MASK;
for (int32_t i = 1; i < tx_buff_length + 1; i++) {
2013-08-27 15:52:20 +02:00
if (i < I2C_BUFSIZE) {
i2c_master_buffer[i] = tx_buff[i - 1];
}
}
//enable I2C to read
if ((rx_buff_length > 0) && (i < I2C_BUFSIZE)) {
i2c_master_buffer[i] = ((slave_addr << 1) & WRITE_ENABLE_BIT_MASK)
| READ_ENABLE_BIT_MASK;
read_index = i + 1;
}
bool successful = i2c_transaction(i2c_interface);
2013-08-27 15:52:20 +02:00
if (successful && (rx_buff != NULL) && (rx_buff_length > 0)) {
memcpy(rx_buff, (const uint8_t *)(i2c_master_buffer + read_index),
sizeof(uint8_t) * rx_buff_length);
return true;
}
else {
return false;
}
}
else {
puts(
"[i2c.c/i2cRead]: the txBuff is not valid or has not a valid \
length value !\n");
return false;
}
2013-08-07 16:54:10 +02:00
}
/**
2013-08-27 15:52:20 +02:00
* @brief The interrupt handler for the I2C0 interface.
* It deals only with the master mode.
2013-08-07 16:54:10 +02:00
*
*/
void i2c_interface0_master_handler(void) //__irq
{
2013-08-27 15:52:20 +02:00
//puts("entering I2C handler function\n");
uint8_t state_value;
/* this handler deals with master read and master write only */
state_value = I20STAT;
//IENABLE; /* handles nested interrupt */
//enableIRQ();
switch (state_value) {
case 0x08: /* A Start condition is issued. */
//puts("A Start condition is issued\n");
I20DAT = i2c_master_buffer[0];
//printf("I22DAT = %lu\n", I22DAT);
I20CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_STARTED;
break;
case 0x10: /* A repeated started is issued */
//puts("A repeated Start is issued\n");
// if ( I2CCmd == L3DG420_WHO_AM_I)
// {
// I22DAT = I2CMasterBuffer[2];
// }
I20DAT = i2c_master_buffer[2];
I20CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_RESTARTED;
break;
case 0x18: /* Regardless, it's a ACK */
//puts("got an Ack\n");
if (i2c_master_state == I2C_STARTED) {
I20DAT = i2c_master_buffer[1 + wr_index];
wr_index++;
i2c_master_state = DATA_ACK;
}
I20CONCLR = I2CONCLR_SIC;
break;
case 0x28: /* Data byte has been transmitted, regardless ACK or NACK */
case 0x30:
//puts("Data byte has been transmitted\n");
if (wr_index != i2c_write_length) {
// this should be the last one
I20DAT = i2c_master_buffer[1 + wr_index];
if (wr_index != i2c_write_length) {
i2c_master_state = DATA_ACK;
}
else {
i2c_master_state = DATA_NACK;
I20CONSET = I2CONSET_STO; /* Set Stop flag */
if (i2c_read_length != 0) {
I20CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
}
wr_index++;
}
else {
if (i2c_read_length != 0) {
I20CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
else {
i2c_master_state = DATA_NACK;
I20CONSET = I2CONSET_STO; /* Set Stop flag */
}
}
I20CONCLR = I2CONCLR_SIC;
break;
case 0x40: /* Master Receive, SLA_R has been sent */
//puts("Master Receive, SLA_R has been sent!\n");
if (i2c_read_length >= 2) {
I20CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
I20CONCLR = I2CONCLR_SIC;
break;
// Data byte has been received, regardless following ACK or NACK
case 0x50:
case 0x58:
//puts("Data received\n");
i2c_master_buffer[3 + rd_index] = I20DAT;
rd_index++;
if (rd_index < (i2c_read_length - 1)) {
i2c_master_state = DATA_ACK;
I20CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
else {
I20CONCLR = I2CONCLR_AAC; /* NACK after data is received */
}
if (rd_index == i2c_read_length) {
rd_index = 0;
i2c_master_state = DATA_NACK;
}
I20CONCLR = I2CONCLR_SIC;
break;
case 0x20: /* regardless, it's a NACK */
case 0x48:
I20CONCLR = I2CONCLR_SIC;
i2c_master_state = DATA_NACK;
break;
case 0x38: /*
* Arbitration lost, in this example, we don't
* deal with multiple master situation
**/
//puts("Arbritration lost!\n");
default:
I20CONCLR = I2CONCLR_SIC;
break;
}
//IDISABLE;
//disableIRQ();
//puts("leave I2C handler function\n");
VICVectAddr = 0; /* Acknowledge Interrupt */
2013-08-07 16:54:10 +02:00
}
/**
2013-08-27 15:52:20 +02:00
* @brief The interrupt handler for the I2C1 interface.
* It deals only with the master mode.
2013-08-07 16:54:10 +02:00
*
*/
void i2c_interface1_master_handler(void) //__irq
{
2013-08-27 15:52:20 +02:00
//puts("entering I2C handler function\n");
uint8_t state_value;
/* this handler deals with master read and master write only */
state_value = I21STAT;
//IENABLE; /* handles nested interrupt */
//enableIRQ();
switch (state_value) {
case 0x08: /* A Start condition is issued. */
//puts("A Start condition is issued\n");
I21DAT = i2c_master_buffer[0];
//printf("I22DAT = %lu\n", I22DAT);
I21CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_STARTED;
break;
case 0x10: /* A repeated started is issued */
//puts("A repeated Start is issued\n");
// if ( I2CCmd == L3DG420_WHO_AM_I)
// {
// I22DAT = I2CMasterBuffer[2];
// }
I21DAT = i2c_master_buffer[2];
I21CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_RESTARTED;
break;
case 0x18: /* Regardless, it's a ACK */
//puts("got an Ack\n");
if (i2c_master_state == I2C_STARTED) {
I21DAT = i2c_master_buffer[1 + wr_index];
wr_index++;
i2c_master_state = DATA_ACK;
}
I21CONCLR = I2CONCLR_SIC;
break;
case 0x28: /* Data byte has been transmitted, regardless ACK or NACK */
case 0x30:
//puts("Data byte has been transmitted\n");
if (wr_index != i2c_write_length) {
// this should be the last one
I21DAT = i2c_master_buffer[1 + wr_index];
if (wr_index != i2c_write_length) {
i2c_master_state = DATA_ACK;
}
else {
i2c_master_state = DATA_NACK;
I21CONSET = I2CONSET_STO; /* Set Stop flag */
if (i2c_read_length != 0) {
I21CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
}
wr_index++;
}
else {
if (i2c_read_length != 0) {
I21CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
else {
i2c_master_state = DATA_NACK;
I21CONSET = I2CONSET_STO; /* Set Stop flag */
}
}
I21CONCLR = I2CONCLR_SIC;
break;
case 0x40: /* Master Receive, SLA_R has been sent */
//puts("Master Receive, SLA_R has been sent!\n");
if (i2c_read_length >= 2) {
I21CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
I21CONCLR = I2CONCLR_SIC;
break;
case 0x50: /*
* Data byte has been received, regardless following ACK or
* NACK
**/
case 0x58:
//puts("Data received\n");
i2c_master_buffer[3 + rd_index] = I21DAT;
rd_index++;
if (rd_index < (i2c_read_length - 1)) {
i2c_master_state = DATA_ACK;
I21CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
else {
I21CONCLR = I2CONCLR_AAC; /* NACK after data is received */
}
if (rd_index == i2c_read_length) {
rd_index = 0;
i2c_master_state = DATA_NACK;
}
I21CONCLR = I2CONCLR_SIC;
break;
case 0x20: /* regardless, it's a NACK */
case 0x48:
I21CONCLR = I2CONCLR_SIC;
i2c_master_state = DATA_NACK;
break;
case 0x38: /*
* Arbitration lost, in this example, we don't
* deal with multiple master situation
**/
//puts("Arbritration lost!\n");
default:
I21CONCLR = I2CONCLR_SIC;
break;
}
//IDISABLE;
//disableIRQ();
//puts("leave I2C handler function\n");
VICVectAddr = 0; /* Acknowledge Interrupt */
2013-08-07 16:54:10 +02:00
}
/**
2013-08-27 15:52:20 +02:00
* @brief The interrupt handler for the I2C2 interface.
* It deals only with the master mode.
2013-08-07 16:54:10 +02:00
*
*/
2013-08-27 15:52:20 +02:00
void i2c_interface2_master_handler(void) //__irq
2013-08-07 16:54:10 +02:00
{
2013-08-27 15:52:20 +02:00
//puts("entering I2C handler function\n");
uint8_t state_value;
/* this handler deals with master read and master write only */
state_value = I22STAT;
//IENABLE; /* handles nested interrupt */
//enableIRQ();
switch (state_value) {
case 0x08: /* A Start condition is issued. */
//puts("A Start condition is issued\n");
I22DAT = i2c_master_buffer[0];
//printf("I22DAT = %lu\n", I22DAT);
I22CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_STARTED;
break;
case 0x10: /* A repeated started is issued */
//puts("A repeated Start is issued\n");
// if ( I2CCmd == L3DG420_WHO_AM_I)
// {
// I22DAT = I2CMasterBuffer[2];
// }
I22DAT = i2c_master_buffer[2];
I22CONCLR = (I2CONCLR_SIC | I2CONCLR_STAC);
i2c_master_state = I2C_RESTARTED;
break;
case 0x18: /* Regardless, it's a ACK */
//puts("got an Ack\n");
if (i2c_master_state == I2C_STARTED) {
I22DAT = i2c_master_buffer[1 + wr_index];
wr_index++;
i2c_master_state = DATA_ACK;
}
I22CONCLR = I2CONCLR_SIC;
break;
case 0x28: /* Data byte has been transmitted, regardless ACK or NACK */
case 0x30:
//puts("Data byte has been transmitted\n");
if (wr_index != i2c_write_length) {
// this should be the last one
I22DAT = i2c_master_buffer[1 + wr_index];
if (wr_index != i2c_write_length) {
i2c_master_state = DATA_ACK;
}
else {
i2c_master_state = DATA_NACK;
I22CONSET = I2CONSET_STO; /* Set Stop flag */
if (i2c_read_length != 0) {
I22CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
}
wr_index++;
}
else {
if (i2c_read_length != 0) {
I22CONSET = I2CONSET_STA; /* Set Repeated-start flag */
i2c_master_state = I2C_REPEATED_START;
}
else {
i2c_master_state = DATA_NACK;
I22CONSET = I2CONSET_STO; /* Set Stop flag */
}
}
I22CONCLR = I2CONCLR_SIC;
break;
case 0x40: /* Master Receive, SLA_R has been sent */
//puts("Master Receive, SLA_R has been sent!\n");
if (i2c_read_length >= 2) {
I22CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
I22CONCLR = I2CONCLR_SIC;
break;
case 0x50: /*
* Data byte has been received, regardless following ACK or
* NACK
**/
case 0x58:
//puts("Data received\n");
i2c_master_buffer[3 + rd_index] = I22DAT;
rd_index++;
if (rd_index < (i2c_read_length - 1)) {
i2c_master_state = DATA_ACK;
I22CONSET = I2CONSET_AA; /* assert ACK after data is received */
}
else {
I22CONCLR = I2CONCLR_AAC; /* NACK after data is received */
}
if (rd_index == i2c_read_length) {
rd_index = 0;
i2c_master_state = DATA_NACK;
}
I22CONCLR = I2CONCLR_SIC;
break;
case 0x20: /* regardless, it's a NACK */
case 0x48:
I22CONCLR = I2CONCLR_SIC;
i2c_master_state = DATA_NACK;
break;
case 0x38: /*
* Arbitration lost, in this example, we don't
* deal with multiple master situation
**/
//puts("Arbritration lost!\n");
default:
I22CONCLR = I2CONCLR_SIC;
break;
}
//IDISABLE;
//disableIRQ();
//puts("leave I2C handler function\n");
VICVectAddr = 0; /* Acknowledge Interrupt */
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
void i2c_set_slave_mode(uint8_t i2c_interface, uint8_t slave_addr)
{
switch (i2c_interface) {
case I2C0:
I20ADR = slave_addr;
break;
case I2C1_0:
case I2C1_1:
I21ADR = slave_addr;
break;
case I2C2:
I22ADR = slave_addr;
}
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
void i2c_enable_pull_up_resistor(uint8_t i2c_interface)
{
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
switch (i2c_interface) {
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
case I2C1_0: // P0.0 SDA1, P0.1 SCL1
puts("The on-chip pull-up resistor is enabled for the I2C1_0");
PINMODE0 &= ~(BIT0 | BIT1 | BIT2 | BIT3);
break;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
case I2C1_1: // P0.19 SDA1, P0.20 SCL1
puts("The on-chip pull-up resistor is enabled for the I2C1_1");
PINMODE1 &= ~(BIT6 | BIT7 | BIT8 | BIT9);
break;
2013-08-07 16:54:10 +02:00
2013-08-27 15:52:20 +02:00
case I2C2: // P0.10 SDA2, P0.11 SCL2
puts("The on-chip pull-up resistor is enabled for the I2C2");
PINMODE0 &= ~(BIT20 | BIT21 | BIT22 | BIT23);
//PINMODE0 &= ~(BIT20 | BIT22);
}
2013-08-07 16:54:10 +02:00
}
2013-08-27 15:52:20 +02:00
void i2c_disable_pull_up_resistor(uint8_t i2c_interface)
{
switch (i2c_interface) {
case I2C1_0: // P0.0 SDA1, P0.1 SCL1
puts("The on-chip pull-up resistor is disbled for the I2C1_0");
PINMODE0 &= ~(BIT0 | BIT2);
PINMODE0 |= (BIT1 | BIT3);
break;
case I2C1_1: // P0.19 SDA1, P0.20 SCL1
puts("The on-chip pull-up resistor is disabled for the I2C1_1");
PINMODE1 &= ~(BIT6 | BIT8);
PINMODE1 |= (BIT7 | BIT9);
break;
case I2C2: // P0.10 SDA2, P0.11 SCL2
puts("The on-chip pull-up resistor is disbled for the I2C2");
PINMODE0 &= ~(BIT20 | BIT22);
PINMODE1 |= (BIT21 | BIT23);
}
2013-08-07 16:54:10 +02:00
}