реклама на сайте
подробности

 
 
> Проблема с SPI.
Jenya7
сообщение Jun 21 2018, 13:17
Сообщение #1


Профессионал
*****

Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075



я работаю с MK10FN1M0xxx12
посылаю-принимаю по СПИ
Код
uint32_t SPI_TransferByte(uint32_t spi_num, uint8_t tx_data, uint8_t *rx_data)
{
     uint32_t timeout;
     uint32_t value;


    // Assert CS0, Use config 0
   SPIx[spi_num]->PUSHR = SPI_PUSHR_CTAS(0) | SPI_PUSHR_TXDATA((uint32_t)tx_data);

    timeout = 0;
    // while transfer complete
    while (!(SPIx[spi_num]->SR & SPI_SR_TCF_MASK))
    {
       timeout++;
       if (timeout > SPI_TIMEOUT)
       {
         // clear flag
         SPIx[spi_num]->SR = SPI_SR_TCF_MASK;
         return SPI_TIMEOUT_ERROR;
       }
    };

    // clear flag
    SPIx[spi_num]->SR = SPI_SR_TCF_MASK;
    *rx_data = (uint8_t)SPIx[spi_num]->POPR;

    return 0;
}
вижу фигня какя то - раз получил байт в ответ, раз не получил. а на скопе на пине SIN все четко, байты приходят. думал-гадал - добавил задержку
Код
uint32_t SPI_TransferByte(uint32_t spi_num, uint8_t tx_data, uint8_t *rx_data)
{
     uint32_t timeout;
     uint32_t value;


    // Assert CS0, Use config 0
   SPIx[spi_num]->PUSHR = SPI_PUSHR_CTAS(0) | SPI_PUSHR_TXDATA((uint32_t)tx_data);

    timeout = 0;
    // while transfer complete
    while (!(SPIx[spi_num]->SR & SPI_SR_TCF_MASK))
    {
       timeout++;
       if (timeout > SPI_TIMEOUT)
       {
         // clear flag
         SPIx[spi_num]->SR = SPI_SR_TCF_MASK;
         return SPI_TIMEOUT_ERROR;
       }
    };

    // clear flag
    SPIx[spi_num]->SR = SPI_SR_TCF_MASK;

  for (timeout = 0; timeout < 100; timeout++);

    //while (!(SPIx[spi_num]->SR & SPI_SR_RFDF_MASK))  //dosen't work

    *rx_data = (uint8_t)SPIx[spi_num]->POPR;

    return 0;
}
и все работает как часики.
фига се думаю - байт ушел но нужно ждать пока пришедший байт готов. но не задержками же.
начал перебирать флаги в статусном регистре - этот вроде подходит - while (!(SPIx[spi_num]->SR & SPI_SR_RFDF_MASK)) но я не выхожу из while. Че там твориться вообще с Кинетисовским СПИ? Или я что то не так настроил?


Сообщение отредактировал Jenya7 - Jun 21 2018, 15:01
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов
Obam
сообщение Jun 21 2018, 16:17
Сообщение #2


Знающий
****

Группа: Участник
Сообщений: 756
Регистрация: 14-11-14
Пользователь №: 83 663



Тут, судя по ветке про таймер и по "...перебирать флаги в статусном регистре...", проблема в прочитать RefMan.

Я уже почти влюбился wink.gif (в самом хрошем смысле этого слова) в K10.


--------------------
Пролетарий умственного труда.
Go to the top of the page
 
+Quote Post
AlexandrY
сообщение Jun 21 2018, 20:38
Сообщение #3


Ally
******

Группа: Модераторы
Сообщений: 6 232
Регистрация: 19-01-05
Пользователь №: 2 050



Цитата(Obam @ Jun 21 2018, 19:17) *
Тут, судя по ветке про таймер и по "...перебирать флаги в статусном регистре...", проблема в прочитать RefMan.

В Kinetis все операции с SPI делаю через DMA. Тогда получается надежно.
А без DMA, да, есть такая проблема, как чтение раньше времени или слишком поздно. laughing.gif
Go to the top of the page
 
+Quote Post
Jenya7
сообщение Jun 22 2018, 10:43
Сообщение #4


Профессионал
*****

Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075



Цитата(AlexandrY @ Jun 22 2018, 01:38) *
В Kinetis все операции с SPI делаю через DMA. Тогда получается надежно.
А без DMA, да, есть такая проблема, как чтение раньше времени или слишком поздно. laughing.gif


Александр, Вам как главному специалисту по Кинетису вопросы
1. почему флаг while (!(SPIx[spi_num]->SR & SPI_SR_RFDF_MASK)) не работает?
2. Можете показать пример с DMA? У меня не заработало.

я вообще в шоке от Кинетивского СПИ. что это за FIFO в котором нельзя настроить глубину. где флаг дата пришла в POPR. я уже несколько дней танцы с бубнами устраиваю.
пока что единственное решение которое работает for (timeout = 0; timeout < 10; timeout++); перед тем как читать из POPR.
STMовский СПИ просто мерседес в сравнении с Кинетисовским.

Сообщение отредактировал Jenya7 - Jun 22 2018, 11:09
Go to the top of the page
 
+Quote Post
AlexandrY
сообщение Jun 22 2018, 11:28
Сообщение #5


Ally
******

Группа: Модераторы
Сообщений: 6 232
Регистрация: 19-01-05
Пользователь №: 2 050



Цитата(Jenya7 @ Jun 22 2018, 13:43) *
Александр, Вам как главному специалисту по Кинетису вопросы
1. почему флаг while (!(SPIx[spi_num]->SR & SPI_SR_RFDF_MASK)) не работает?
2. Можете показать пример с DMA? У меня не заработало.

я вообще в шоке от Кинетивского СПИ. что это за FIFO в котором нельзя настроить глубину. где флаг дата пришла в POPR. я уже несколько дней танцы с бубнами устраиваю.
пока что единственное решение которое работает for (timeout = 0; timeout < 10; timeout++); перед тем как читать из POPR.
STMовский СПИ просто мерседес в сравнении с Кинетисовским.


Работу с SPI можно посмотреть в моем проекте - https://github.com/Indemsys/Universal3PHalf.../K66BLEZ1_SPI.c
В Kinetis у разных SPI разная глубина и чаще всего она равна 1, увы.
С флагом SPI_SR_RFDF_MASK у меня тоже что-то не поучалось. Уж не помню что именно.
Попробуйте перед приемо-передачей очищать FIFO.
Код
void SPI_clear_FIFO(uint8_t modn)
{
  SPI_Type  *SPI   = spi_mods[modn].spi;

  SPI->MCR |= BIT(HALT); // Остановить модуль
  while (SPI->SR & BIT(TXRXS)); // Ожидание пока бит 30 (TXRXS) не станет равным нулю

  SPI->SR = SPI->SR;
  SPI->MCR |= BIT(CLR_TXF) + BIT(CLR_RXF); // Установить флаги CLR_TXF и CLR_RXF для очистки обоих FIFO
  SPI->MCR &= ~BIT(HALT); // Запустить модуль
}


Go to the top of the page
 
+Quote Post



Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 13th May 2024 - 19:51
Рейтинг@Mail.ru


Страница сгенерированна за 0.01417 секунд с 7
ELECTRONIX ©2004-2016