2010-12-06 12:05:03 +01:00
|
|
|
#include <irq.h>
|
2010-12-11 12:09:20 +01:00
|
|
|
#include <cc110x-defaultSettings.h>
|
|
|
|
#include <cc110x-reg.h>
|
2010-12-06 12:05:03 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <board.h>
|
|
|
|
#include <hwtimer.h>
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
/**************************************************************************************************
|
|
|
|
* @fn Strobe
|
|
|
|
* @brief Send command to radio.
|
|
|
|
* @param none
|
|
|
|
* @return none
|
|
|
|
*************************************************************************************************/
|
|
|
|
uint8_t cc110x_strobe(uint8_t c)
|
|
|
|
{
|
|
|
|
uint8_t statusByte = 0;
|
|
|
|
uint16_t int_state, gdo_state;
|
|
|
|
|
|
|
|
/* Check for valid strobe command */
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((c == 0xBD) || ((c > RF_SRES) && (c < RF_SNOP))) {
|
2013-06-21 03:52:57 +02:00
|
|
|
int_state = disableIRQ();
|
|
|
|
|
|
|
|
/* Clear the Status read flag */
|
|
|
|
RF1AIFCTL1 &= ~(RFSTATIFG);
|
|
|
|
|
|
|
|
/* Wait for radio to be ready for next instruction */
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFINSTRIFG));
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
/* Write the strobe instruction */
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((c > RF_SRES) && (c < RF_SNOP)) {
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
gdo_state = cc110x_read_reg(IOCFG2); /* buffer IOCFG2 state */
|
|
|
|
cc110x_write_reg(IOCFG2, 0x29); /* c-ready to GDO2 */
|
|
|
|
|
|
|
|
RF1AINSTRB = c;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if ((RF1AIN & 0x04) == 0x04) { /* chip at sleep mode */
|
|
|
|
if ((c == RF_SXOFF) || (c == RF_SPWD) || (c == RF_SWOR)) { }
|
2013-06-21 03:52:57 +02:00
|
|
|
else {
|
2013-06-24 22:37:35 +02:00
|
|
|
while ((RF1AIN & 0x04) == 0x04); /* c-ready ? */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
hwtimer_wait(RTIMER_TICKS(9800)); /* Delay for ~810usec at 12MHz CPU clock */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cc110x_write_reg(IOCFG2, gdo_state); /* restore IOCFG2 setting */
|
|
|
|
}
|
|
|
|
else { /* chip active mode */
|
|
|
|
RF1AINSTRB = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
statusByte = RF1ASTATB;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFSTATIFG));
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
return statusByte;
|
2010-12-06 12:05:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
/***************************************************************************************************
|
|
|
|
* @fn cc110x_read_reg
|
|
|
|
* @brief Read byte from register.
|
|
|
|
* @param none
|
|
|
|
* @return none
|
|
|
|
***************************************************************************************************/
|
|
|
|
uint8_t cc110x_read_reg(uint8_t addr)
|
|
|
|
{
|
|
|
|
unsigned char x;
|
|
|
|
uint16_t int_state;
|
2010-12-06 12:05:03 +01:00
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
int_state = disableIRQ();
|
2010-12-06 12:05:03 +01:00
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
RF1AINSTR1B = (addr | RF_REGRD);
|
|
|
|
x = RF1ADOUT1B;
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
2013-06-21 03:52:57 +02:00
|
|
|
return x;
|
2010-12-06 12:05:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
/**************************************************************************************************
|
|
|
|
* @fn cc110x_write_reg
|
|
|
|
* @brief Write byte to register.
|
|
|
|
* @param none
|
|
|
|
* @return none
|
|
|
|
**************************************************************************************************/
|
|
|
|
void cc110x_write_reg(uint8_t addr, uint8_t value)
|
|
|
|
{
|
|
|
|
volatile unsigned int i;
|
|
|
|
uint16_t int_state;
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
int_state = disableIRQ();
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFINSTRIFG)); /* Wait for the Radio to be ready for the next instruction */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
RF1AINSTRW = ((addr | RF_REGWR) << 8) + value; /* Send address + Instruction */
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RFDINIFG & RF1AIFCTL1));
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
i = RF1ADOUTB; /* Reset RFDOUTIFG flag which contains status byte */
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
uint8_t cc110x_read_status(uint8_t addr)
|
|
|
|
{
|
2010-12-10 18:00:31 +01:00
|
|
|
unsigned char x;
|
|
|
|
uint16_t int_state;
|
|
|
|
|
|
|
|
int_state = disableIRQ();
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
RF1AINSTR1B = (addr | RF_STATREGRD);
|
2010-12-10 18:00:31 +01:00
|
|
|
x = RF1ADOUT1B;
|
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
|
|
|
return x;
|
2010-12-06 12:05:03 +01:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
/****************************************************************************************************
|
|
|
|
* @fn cc110x_readburst_reg
|
|
|
|
* @brief Read sequence of bytes from register.
|
|
|
|
* @param none
|
|
|
|
* @return none
|
|
|
|
***************************************************************************************************/
|
|
|
|
void cc110x_readburst_reg(uint8_t addr, char *buffer, uint8_t count)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
uint16_t int_state;
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
int_state = disableIRQ();
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFINSTRIFG)); /* Wait for the Radio to be ready for next instruction */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
RF1AINSTR1B = (addr | RF_REGRD); /* Send address + Instruction */
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (i = 0; i < (count - 1); i++) {
|
|
|
|
while (!(RFDOUTIFG & RF1AIFCTL1)); /* Wait for the Radio Core to update the RF1ADOUTB reg */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
buffer[i] = RF1ADOUT1B; /* Read DOUT from Radio Core + clears RFDOUTIFG */
|
|
|
|
/* Also initiates auo-read for next DOUT byte */
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer[count - 1] = RF1ADOUT0B; /* Store the last DOUT from Radio Core */
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
2013-06-21 03:52:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void cc110x_read_fifo(char *buffer, uint8_t count)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
uint16_t int_state;
|
2010-12-11 12:09:20 +01:00
|
|
|
|
|
|
|
int_state = disableIRQ();
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFINSTRIFG)); /* Wait for the Radio to be ready for next instruction */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
RF1AINSTR1B = (RF_RXFIFORD); /* Send address + Instruction */
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (i = 0; i < (count - 1); i++) {
|
|
|
|
while (!(RFDOUTIFG & RF1AIFCTL1)); /* Wait for the Radio Core to update the RF1ADOUTB reg */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
buffer[i] = RF1ADOUT1B; /* Read DOUT from Radio Core + clears RFDOUTIFG */
|
|
|
|
/* Also initiates auo-read for next DOUT byte */
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer[count - 1] = RF1ADOUT0B; /* Store the last DOUT from Radio Core */
|
2010-12-06 12:05:03 +01:00
|
|
|
|
2010-12-11 12:09:20 +01:00
|
|
|
restoreIRQ(int_state);
|
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
/***************************************************************************************************
|
|
|
|
* @fn cc110x_writeburst_reg
|
|
|
|
* @brief Write sequence of bytes to register.
|
|
|
|
* @param none
|
|
|
|
* @return none
|
|
|
|
**************************************************************************************************/
|
|
|
|
uint8_t cc110x_writeburst_reg(uint8_t addr, char *buffer, uint8_t count)
|
|
|
|
{
|
|
|
|
/* Write Burst works wordwise not bytewise - bug known already */
|
|
|
|
unsigned char i;
|
|
|
|
uint16_t int_state;
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
int_state = disableIRQ();
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RF1AIFCTL1 & RFINSTRIFG)); /* Wait for the Radio to be ready for next instruction */
|
2013-06-21 03:52:57 +02:00
|
|
|
|
|
|
|
RF1AINSTRW = ((addr | RF_REGWR) << 8) + buffer[0]; /* Send address + Instruction */
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (i = 1; i < count; i++) {
|
2013-06-21 03:52:57 +02:00
|
|
|
RF1ADINB = buffer[i]; /* Send data */
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (!(RFDINIFG & RF1AIFCTL1)); /* Wait for TX to finish */
|
2013-06-21 03:52:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
i = RF1ADOUTB; /* Reset RFDOUTIFG flag which contains status byte */
|
2010-12-06 12:05:03 +01:00
|
|
|
|
|
|
|
restoreIRQ(int_state);
|
|
|
|
return count;
|
|
|
|
}
|