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

 
 
 
Reply to this topicStart new topic
> SPI & DMA на UC3, либо twi
altlogic
сообщение Jun 9 2009, 09:58
Сообщение #1


Местный
***

Группа: Свой
Сообщений: 222
Регистрация: 2-12-06
Из: г. Хабаровск
Пользователь №: 23 035



Здравствуйте!

Собственно делал ли кто-нибудь связку интерфейса SPI (TWI) с PDCA? Моя реализация работала, но через ... До поры до времени. Теперь нужно сделать всё правильноsmile.gif Проблема с чтением dummy- байтов по SPI.


--------------------
С уважением, Вячеслав
Go to the top of the page
 
+Quote Post
vibro
сообщение Jun 11 2009, 07:35
Сообщение #2





Группа: Участник
Сообщений: 9
Регистрация: 28-04-09
Пользователь №: 48 380



Да, делал.

Чтобы что-то считать с SPI, нужно что-нибудь записать в него. Т.е. отправляешь через один канал DMA буфер в SPI, по второму каналу читаешь буфер из SPI.

Сообщение отредактировал vibro - Jun 11 2009, 07:36


--------------------
Go to the top of the page
 
+Quote Post
altlogic
сообщение Jun 13 2009, 04:38
Сообщение #3


Местный
***

Группа: Свой
Сообщений: 222
Регистрация: 2-12-06
Из: г. Хабаровск
Пользователь №: 23 035



Да, я знаю что нужно передавать так называемые dummy-байты. но у меня возникла проблема с чтением. Для чтения данных я вынужден инициализировать канал приёма данных на число элементов N+1, где N - количество принимаемых байт данных.
Код приёмника у меня получился такой:
Код
pdca_load_channel(PDCA_CHANEL_NANO_SPI_RX, ((unsigned char *)DataStr)-1, anz+1);
pdca_load_channel(PDCA_CHANEL_NANO_SPI_TX, (unsigned char *)ffbyte, anz);

pdca_enable(PDCA_CHANEL_NANO_SPI_RX);  // получить данные
pdca_enable(PDCA_CHANEL_NANO_SPI_TX);  // Передать dummy - байты

// Дождаться окончания передачи по каналу DMA
while (pdca_get_load_size(PDCA_CHANEL_NANO_SPI_TX));
pdca_disable(PDCA_CHANEL_NANO_SPI_TX);  // Disable now the transfer.
pdca_disable(PDCA_CHANEL_NANO_SPI_RX);  // Disable now the transfer.

Всё работало хорошо до тех пор пока я не обнаружил странный баг. Один байт данных в оперативной памяти повреждается после такого вот приёма через DMA. Байт принимает значение 0xff. По всей видимости тот самый лишний байт и пишется туда, куда я не хочу. Если загружать канал DMA на приём без смещения и на длину равную длине пересылки, то принимается количество данных меньшее, чем указано и со смещением. Есть туманные предположения, что это связано с тем, что последний байт из приёмника данных нужно считывать как-то особенно.


--------------------
С уважением, Вячеслав
Go to the top of the page
 
+Quote Post
aaarrr
сообщение Jun 13 2009, 07:16
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(altlogic @ Jun 13 2009, 08:38) *
Есть туманные предположения, что это связано с тем, что последний байт из приёмника данных нужно считывать как-то особенно.

Туман надо разогнать:
- передача заканчивается не тогда, когда у PDC TX кончатся данные, а когда эти данные кончатся и в сдвиговом регистре SPI
- если RHR на момент начала передачи полон (например, использовалась только передача без приема), то его нужно предварительно очистить чтением
Go to the top of the page
 
+Quote Post
altlogic
сообщение Jun 13 2009, 09:06
Сообщение #5


Местный
***

Группа: Свой
Сообщений: 222
Регистрация: 2-12-06
Из: г. Хабаровск
Пользователь №: 23 035



Цитата(aaarrr @ Jun 13 2009, 18:16) *
Туман надо разогнать:
- передача заканчивается не тогда, когда у PDC TX кончатся данные, а когда эти данные кончатся и в сдвиговом регистре SPI

Поставил проверку на наличие данных в сдвиговом регистре вслед за проверкой данных у PDC TX, и всё равно приём идёт некорректно...
Цитата(aaarrr @ Jun 13 2009, 18:16) *
- если RHR на момент начала передачи полон (например, использовалась только передача без приема), то его нужно предварительно очистить чтением

У SPI нет такого регистра. Аналогичный ему регистр Receive Data Register не очищается чтением.


--------------------
С уважением, Вячеслав
Go to the top of the page
 
+Quote Post
aaarrr
сообщение Jun 13 2009, 09:11
Сообщение #6


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(altlogic @ Jun 13 2009, 13:06) *
Поставил проверку на наличие данных в сдвиговом регистре вслед за проверкой данных у PDC TX, и всё равно приём идёт некорректно...

Код приведите.

Цитата(altlogic @ Jun 13 2009, 13:06) *
У SPI нет такого регистра. Аналогичный ему регистр Receive Data Register не очищается чтением.

Что значит не очищается? RDRF в SR не сбрасывается? Не верю.
Go to the top of the page
 
+Quote Post
altlogic
сообщение Jun 14 2009, 23:21
Сообщение #7


Местный
***

Группа: Свой
Сообщений: 222
Регистрация: 2-12-06
Из: г. Хабаровск
Пользователь №: 23 035



Решение я нашёл, это радует. НО нашёл его практически методом тыкаsmile.gif Хотелось бы разобраться в сути. Кода очищения буфера spi не осталось. После предварительного dma- чтения без передачи всё нормализовалось.


--------------------
С уважением, Вячеслав
Go to the top of the page
 
+Quote Post
aaarrr
сообщение Jun 15 2009, 07:25
Сообщение #8


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(altlogic @ Jun 15 2009, 03:21) *
Кода очищения буфера spi не осталось. После предварительного dma- чтения без передачи всё нормализовалось.

Чем "предварительное DMA-чтение без передачи" отличается от "очищения буфера SPI"?
Никаких "хитростей", кроме уже упомянутых, в работе с SPI по DMA нет - так что осталось не понятным?
Go to the top of the page
 
+Quote Post
altlogic
сообщение Jun 15 2009, 09:20
Сообщение #9


Местный
***

Группа: Свой
Сообщений: 222
Регистрация: 2-12-06
Из: г. Хабаровск
Пользователь №: 23 035



Вот отрывок кода предварительного чтения dma-канала:
Код
pdca_channel->mar = (unsigned long)addr;          //загрузка указателя данных
pdca_channel->tcr = size;                                  // длина передачи
pdca_channel->cr = AVR32_PDCA_TEN_MASK;    // задействовать канал приёма
pdca_channel->cr = AVR32_PDCA_TDIS_MASK;  // отключить канал приёма

Как-то не очень похоже на очищение буфера spi, видимо таким образом очищается буфер dma (какой ещё буфер???) Пока не прочитаю в даташите, что это так, не поверю... В описании регистра PDCA Control Register биты TransferEnable и TransferDisable (AVR32_PDCA_TEN_MASK и AVR32_PDCA_TDIS_MASK в коде) соответственно включают и выключают передачу по DMA. Про очитску буферов ничего нет.
Цитата
• TEN: Transfer Enable
0 = No Effect.
1 = Enable transfer for DMA channel.
• TDIS: Transfer Disable
0 = No Effect.
1 = Disable transfer for DMA channel.

Хотя, с другой стороны включение dma-передачи разрешает каналу считать данные из регистра данных приёмника SPI. Но тогда и впрямь простое чтение регистра данных должно быть равносильно активации/декативации канала dma! Попробую всё же ещё раз проверить завтра этот вариант на свежую голову.


--------------------
С уважением, Вячеслав
Go to the top of the page
 
+Quote Post
Storejet
сообщение May 16 2011, 05:54
Сообщение #10





Группа: Участник
Сообщений: 12
Регистрация: 17-03-11
Пользователь №: 63 666



Столкнулся с необходимостью использовать DMA для работы с SPI. Никогда ничего подобного раньше не делал, поэтому не знаю с какой стороны подступиться.Мог бы кто нибудь привести пример кода? SPI у меня в режиме ведущего. Как начать прием данных? Надо писать через DMA dummy байты?
Go to the top of the page
 
+Quote Post
aaarrr
сообщение May 16 2011, 10:54
Сообщение #11


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(Storejet @ May 16 2011, 09:54) *
Надо писать через DMA dummy байты?

Надо. Настраиваете два канала DMA, один на прием, второй на передачу. Запускаете сначала прием, затем передачу.
Go to the top of the page
 
+Quote Post

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

 


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


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