Как передавать и получать данные с использованием SPI с LPC1788
Я пытаюсь написать программу с использованием микроконтроллера LPC1788, и мне нужно отправлять и передавать данные в UFDC (универсальный преобразователь частоты в цифровой формат) с использованием протокола SPI. Таблицу для этого можно найти здесь. Контроллер SSP моего микроконтроллера установлен как ведущий, а UFDC предположительно становится подчиненным. Я могу передать информацию по линии MOSI и увидеть, как данные появляются на осциллографе. Однако, когда я пытаюсь получить данные по линии MISO, я получаю только то, что отправил по MOSI.
Соответствующая часть кода, который я написал, приведена ниже:
SSP_CFG_Type sspConfig;
SSP_DATA_SETUP_Type sspData;
LPC_SSP_TypeDef *SSPx = NULL;
UFDC_RESULT_T result_SSP0_FX1, result_SSP0_FX2;
UFDC_RESULT_T result_SSP1_FX1, result_SSP1_FX2;
UFDC_RESULT_T *result, *resultFX1, *resultFX2 = NULL;
uint8_t resultSign;
uint64_t resultInt, resultFract;
uint8_t SSP0resultFX1sign, SSP0resultFX2sign;
uint8_t SSP1resultFX1sign, SSP1resultFX2sign;
uint64_t SSP0resultFX1int, SSP0resultFX2int;
uint64_t SSP1resultFX1int, SSP1resultFX2int;
uint64_t SSP0resultFX1fract, SSP0resultFX2fract;
uint64_t SSP1resultFX1fract, SSP1resultFX2fract;
uint16_t getAccInstr = 0x01FF;
uint16_t setAccInstr = 0x0204;
uint16_t checkStatusInstr = 0x03FF;
uint16_t setMeasureModeInstr1 = 0x0600;
uint16_t setMeasureModeInstr2 = 0x060E;
uint16_t getBCDResultInstr = 0x07FF;
uint8_t startMeasureInstr = 0x09;
uint32_t measureInstr;
uint8_t txData[2];
uint8_t rxData[2];
uint16_t data;
sspConfig.CPHA = SSP_CPHA_FIRST;
sspConfig.CPOL = SSP_CPOL_HI;
sspConfig.ClockRate = 100000;
sspConfig.Databit = SSP_DATABIT_8;
sspConfig.Mode = SSP_MASTER_MODE;
sspConfig.FrameFormat = SSP_FRAME_SPI;
sspData.tx_data = txData;
sspData.rx_data = rxData;
sspData.length = 2;
printf("Initialising SSP0 and SSP1...\n\n");
PINSEL_ConfigPin(0, 15, 2); // SSP0_SCK
PINSEL_ConfigPin(0, 16, 2); // SSP0_SSEL
PINSEL_ConfigPin(0, 17, 2); // SSP0_MISO
PINSEL_ConfigPin(0, 18, 2); // SSP0_MOSI
PINSEL_ConfigPin(0, 6, 2); // SSP1_SCK
PINSEL_ConfigPin(0, 7, 2); // SSP1_SSEL
PINSEL_ConfigPin(0, 8, 2); // SSP1_MISO
PINSEL_ConfigPin(0, 9, 2); // SSP1_MOSI
PINSEL_SetFilter(0, 7, DISABLE);
PINSEL_SetFilter(0, 8, DISABLE);
PINSEL_SetFilter(0, 9, DISABLE);
SSP_Init(LPC_SSP0, &sspConfig);
SSP_Init(LPC_SSP1, &sspConfig);
SSP_Cmd(LPC_SSP0, ENABLE);
SSP_Cmd(LPC_SSP1, ENABLE);
printf("Reading UDFC frequency values...\n\n");
for(int i=0; i < 2; i++)
{
if(i == 0)
{
SSPx = LPC_SSP0;
resultFX1 = &result_SSP0_FX1;
resultFX2 = &result_SSP0_FX2;
}
else
{
SSPx = LPC_SSP1;
resultFX1 = &result_SSP1_FX1;
resultFX2 = &result_SSP1_FX2;
}
// Set UFDC accuracy to 0.05%.
SSP_SendData(SSPx, (uint8_t)(setAccInstr >> 8));
SSP_SendData(SSPx, (uint8_t)(setAccInstr & 0xFF));
// Check accuracy.
while(1)
{
SSP_SendData(SSPx, (uint8_t)(getAccInstr >> 8));
SSP_SendData(SSPx, (uint8_t)(getAccInstr & 0xFF));
data = SSP_ReceiveData(SSPx);
printf("Accuracy check 1: %i\n", data >> 8);
printf("Accuracy check 2: %i\n", data & 0xFF);
}
// More code follows.
Вывод, который я получаю из этого ниже:
Accuracy check 1: 0
Accuracy check 2: 0
Accuracy check 1: 0
Accuracy check 2: 255
Accuracy check 1: 0
Accuracy check 2: 2
Accuracy check 1: 0
Accuracy check 2: 4
Accuracy check 1: 0
Accuracy check 2: 1
Accuracy check 1: 0
Accuracy check 2: 255
Accuracy check 1: 0
Accuracy check 2: 1
Accuracy check 1: 0
Accuracy check 2: 255
Accuracy check 1: 0
Accuracy check 2: 1
Accuracy check 1: 0
Accuracy check 2: 255
Редактировать: с помощью контроллеров SPI можно установить "возврат по шлейфу", чтобы MOSI считывал входные данные из MISO и наоборот. Тем не менее, я проверил соответствующий регистр, и это отключено.
2 ответа
Я думаю, что вы должны проверить FIFO получения, а не пустой флаг, прямо перед чтением данных с этой строкой кода:
while (! (LPC_SSP0-> SR & SSP_SR_RNE)); // опрос принимает FIFO, а не пустой флаг
и в инициализации SPI вы должны очистить FIFO получения с этими строками кода:
while (LPC_SSP0->SR & SSP_SR_RNE){
tmp = (uint32_t) SSP_ReceiveData(LPC_SSP0);
}
надеюсь, что это сработает!
После отправки одного байта через SPI необходимо снова очистить один байт в регистре данных (DR).
В SPI вы всегда отправляете и получаете байт. Вы не можете просто отправить ИЛИ получить байт, это всегда И. Техническое объяснение состоит в том, что есть только одна тактовая линия и две линии данных (MISO и MOSI)