/** * Called to set the eUSCI module in 'master' mode */ void DWire::_initMaster(const eUSCI_I2C_MasterConfig * i2cConfig) { // Initialise the pins MAP_GPIO_setAsPeripheralModuleFunctionInputPin(modulePort, modulePins, GPIO_PRIMARY_MODULE_FUNCTION); // Initializing I2C Master to SMCLK at 400kbs with no autostop MAP_I2C_initMaster(module, i2cConfig); // Specify slave address MAP_I2C_setSlaveAddress(module, slaveAddress); // Set Master in transmit mode MAP_I2C_setMode(module, EUSCI_B_I2C_TRANSMIT_MODE); // Enable I2C Module to start operations MAP_I2C_enableModule(module); // Enable and clear the interrupt flag MAP_I2C_clearInterruptFlag(module, EUSCI_B_I2C_TRANSMIT_INTERRUPT0 + EUSCI_B_I2C_NAK_INTERRUPT + EUSCI_B_I2C_RECEIVE_INTERRUPT0); // Enable master interrupts MAP_I2C_enableInterrupt(module, EUSCI_B_I2C_TRANSMIT_INTERRUPT0 + EUSCI_B_I2C_NAK_INTERRUPT + EUSCI_B_I2C_RECEIVE_INTERRUPT0); // Register the interrupts on the correct module MAP_Interrupt_enableInterrupt(intModule); MAP_Interrupt_enableMaster(); }
/** * Request data from a SLAVE as a MASTER */ uint8_t DWire::requestFrom(uint_fast8_t slaveAddress, uint_fast8_t numBytes) { // No point of doing anything else if there we're not a MASTER if (busRole != BUS_ROLE_MASTER) return 0; if (*pTxBufferIndex > 0) { endTransmission(false); } while (!sendStop) ; // Re-initialise the rx buffer *pRxBufferSize = numBytes; *pRxBufferIndex = 0; // Configure the correct slave MAP_I2C_setSlaveAddress(module, slaveAddress); this->slaveAddress = slaveAddress; MAP_I2C_disableInterrupt(module, EUSCI_B_I2C_TRANSMIT_INTERRUPT0); // Set the master into receive mode MAP_I2C_setMode(module, EUSCI_B_I2C_RECEIVE_MODE); // Send the START MAP_I2C_masterReceiveStart(module); // Send a stop early if we're only requesting one byte // to prevent timing issues if (numBytes == 1) { MAP_I2C_masterReceiveMultiByteStop(module); } // Initialize the flag showing the status of the request requestDone = false; gotNAK = false; // Wait until the request is done while (!requestDone) ; MAP_I2C_setMode(module, EUSCI_B_I2C_TRANSMIT_MODE); MAP_I2C_enableInterrupt(module, EUSCI_B_I2C_TRANSMIT_INTERRUPT0); MAP_I2C_clearInterruptFlag(module, EUSCI_B_I2C_TRANSMIT_INTERRUPT0); // Reset the buffer (*pRxBufferIndex) = 0; (*pRxBufferSize) = 0; if (gotNAK) { return 0; } else { return rxReadLength; } }
/******************************************************************************* * The USCIAB0TX_ISR is structured such that it can be used to transmit any * number of bytes by pre-loading TXByteCtr with the byte count. Also, TXData * points to the next byte to transmit. *******************************************************************************/ void euscib0_isr(void) { uint_fast16_t status; status = MAP_I2C_getEnabledInterruptStatus(EUSCI_B0_MODULE); MAP_I2C_clearInterruptFlag(EUSCI_B0_MODULE, status); if (status & EUSCI_B_I2C_NAK_INTERRUPT) { MAP_I2C_masterSendStart(EUSCI_B0_MODULE); return; } if (status & EUSCI_B_I2C_TRANSMIT_INTERRUPT0) { MAP_I2C_setSlaveAddress(EUSCI_B0_MODULE, SLAVE_ADDRESS_1); } if (status & EUSCI_B_I2C_TRANSMIT_INTERRUPT1) { MAP_I2C_setSlaveAddress(EUSCI_B0_MODULE, SLAVE_ADDRESS_2); } /* Check the byte counter */ if (TXByteCtr) { /* Send the next data and decrement the byte counter */ MAP_I2C_masterSendMultiByteNext(EUSCI_B0_MODULE, TXData++); TXByteCtr--; } else { MAP_I2C_masterSendMultiByteStop(EUSCI_B0_MODULE); sendStopCondition = true; sendToSlaveAddress2 = !sendToSlaveAddress2; MAP_Interrupt_disableSleepOnIsrExit(); } }
/** * Re-set the slave address (the target address when master or the slave's address when slave) */ void DWire::_setSlaveAddress(uint_fast8_t newAddress) { slaveAddress = newAddress; MAP_I2C_setSlaveAddress(module, newAddress); }
/** * Request data from a SLAVE as a MASTER */ uint8_t DWire::requestFrom( uint_fast8_t slaveAddress, uint_fast8_t numBytes ) { // No point of doing anything else if there we're not a MASTER if (busRole != BUS_ROLE_MASTER) return 0; // still something to send? Flush the TX buffer but do not send a STOP if (*pTxBufferIndex > 0) { // this is a repeated start: no point in trying to receive if we fail finishing the transmission if (endTransmission( false )) { return 0; } } else { // Wait until any request is finished timeout = TIMEOUTLIMIT; while ( MAP_I2C_masterIsStopSent( module ) == EUSCI_B_I2C_SENDING_STOP && timeout) timeout--; } if (!timeout) { /* If we get a timeout, then reset everything */ _resetBus( ); return 0; } // Re-initialise the rx buffer // and make sure we never request 1 byte only // this is an anomalous behaviour of the MSP432 related to the double // buffering of I2C. This is a workaround. if (numBytes == 1) { *pRxBufferSize = 2; } else { *pRxBufferSize = numBytes; } *pRxBufferIndex = 0; // Configure the correct slave MAP_I2C_setSlaveAddress( module, slaveAddress ); this->slaveAddress = slaveAddress; MAP_I2C_clearInterruptFlag( module, EUSCI_B_I2C_RECEIVE_INTERRUPT0 | EUSCI_B_I2C_NAK_INTERRUPT ); MAP_I2C_enableInterrupt( module, EUSCI_B_I2C_RECEIVE_INTERRUPT0 | EUSCI_B_I2C_NAK_INTERRUPT ); // Set the master into receive mode MAP_I2C_setMode( module, EUSCI_B_I2C_RECEIVE_MODE ); // Initialize the flag showing the status of the request requestDone = false; gotNAK = false; // Send the START MAP_I2C_masterReceiveStart( module ); // Wait until the request is done timeout = TIMEOUTLIMIT; while (!requestDone && timeout) timeout--; if (!timeout) { /* If we get a timeout, then reset everything */ _resetBus( ); return 0; } if (gotNAK) { _I2CDelay( ); MAP_I2C_masterReceiveMultiByteStop( module ); return 0; } else { if (numBytes == 1) { return --(*pRxBufferSize); } else { return *pRxBufferSize; } } }
int main(void) { volatile uint32_t ii; /* Disabling the Watchdog */ MAP_WDT_A_holdTimer(); /* Select Port 1 for I2C - Set Pin 6, 7 to input Primary Module Function, * (UCB0SIMO/UCB0SDA, UCB0SOMI/UCB0SCL). */ MAP_GPIO_setAsPeripheralModuleFunctionInputPin(GPIO_PORT_P1, GPIO_PIN6 + GPIO_PIN7, GPIO_PRIMARY_MODULE_FUNCTION); sendStopCondition = false; sendToSlaveAddress2 = false; /* Initializing I2C Master to SMCLK at 400kbs with no autostop */ MAP_I2C_initMaster(EUSCI_B0_MODULE, &i2cConfig); /* Specify slave address. For start we will do our first slave address */ MAP_I2C_setSlaveAddress(EUSCI_B0_MODULE, SLAVE_ADDRESS_1); /* Set Master in receive mode */ MAP_I2C_setMode(EUSCI_B0_MODULE, EUSCI_B_I2C_TRANSMIT_MODE); /* Enable I2C Module to start operations */ MAP_I2C_enableModule(EUSCI_B0_MODULE); /* Enable and clear the interrupt flag */ MAP_I2C_clearInterruptFlag(EUSCI_B0_MODULE, EUSCI_B_I2C_TRANSMIT_INTERRUPT0 + EUSCI_B_I2C_NAK_INTERRUPT); //Enable master Receive interrupt MAP_I2C_enableInterrupt(EUSCI_B0_MODULE, EUSCI_B_I2C_TRANSMIT_INTERRUPT0 + EUSCI_B_I2C_NAK_INTERRUPT); MAP_Interrupt_enableInterrupt(INT_EUSCIB0); while (1) { /* Load Byte Counter */ TXByteCtr = 4; /* Making sure the last transaction has been completely sent out */ while (MAP_I2C_masterIsStopSent(EUSCI_B0_MODULE) == EUSCI_B_I2C_SENDING_STOP); /* Sending the initial start condition for appropriate * slave addresses */ if(sendToSlaveAddress2) { MAP_I2C_setSlaveAddress(EUSCI_B0_MODULE, SLAVE_ADDRESS_2); } else { MAP_I2C_setSlaveAddress(EUSCI_B0_MODULE, SLAVE_ADDRESS_1); } MAP_I2C_masterSendMultiByteStart(EUSCI_B0_MODULE, TXData++); while(sendStopCondition == false) { /* Go to sleep while data is being transmitted */ MAP_Interrupt_enableSleepOnIsrExit(); MAP_PCM_gotoLPM0InterruptSafe(); } sendStopCondition = false; } }