Перейти к содержанию
    

День добрый. Обращаюсь за помощью-подсказкой. Начал изучение интерфейса USB, для этого приобрёл контроллер at90usb162. Решил познать всё с нуля. Проштудировав datasheet, и прочую литературу занялся написание прошивки на языке С в AVR Studio 4. Написав пару программа моргания диодом и ответом на нажатие кнопочки, решил приступить к написанию программы для USB. Сейчас затык на этапе получения пакета SETUP от хоста т.е. ПК. Подскажите, пожалуйста, где ошибка. Код прошивки прилагается.

#include <avr/io.h>
#include <avr/wdt.h>
#include <avr/power.h>


#define F_CPU 16000000UL


#include <util/delay.h>



int main()
{
 DDRC |= 1 << DDC6; // Порт С - как выход

 MCUSR &= ~(1 << WDRF);
 wdt_disable();

 clock_prescale_set(clock_div_1);

 //CLKPR = 0x80;  // Бит разрешение изменения делителя
 //CLKPR = 0x00; // Установка делителя на 1 F = 16 MHz

 PORTC |= (1 << PC6);  // РС6 - зажечь диод

 //REGCR  =  (1 << REGDIS);

 UDIEN = 0x00; //
 UDINT = 0x00; //

 USBCON |=  (1 << USBE);

 USBCON &= ~(1 << FRZCLK);

 PLLCSR |= (1 << PLLP0); // Установить делитель PLL
 if ((PLLCSR & 0x1C) == 0x04)
PORTC = 0x00;  // Потушить диод
 PLLCSR |= (1 << PLLE); // Enable PLL   
 while((PLLCSR & 0x01) != 0x01) // Захват в петле PLL
PORTC = 0x00;  // Потушить диод
 PORTC |= (1<<PC6);  // РС6 - зажечь диод

 /*------Начало настройки нулевой КТ------*/
 UENUM = 0x00; // Нулевая конечая точка
 //UECONX |= 0x29; // Бит разрешения работы КТ + бит сброса переключения данных
 UECONX |= (1 << STALLRQ);
 UECONX |= (1 << EPEN); // Бит разрешения работы КТ + бит сброса переключения данных
 //UECFG0X = 0x81; // Bulk, in
 //UECFG0X = 0x00; // Control, out
 //UECFG0X = 0x01; // Control, in
 //UECFG1X = 0x36; // 64 байта, 2-ой банк, выделение буфера
 UECFG1X = 0x32; // 64 байта, 1 банк, выделение буфера
 //UECFG1X = 0x10; // 16 байт, 1 банк, очистка буфера
 while ((UESTA0X & 0x80) != 0x80) // Проверка корректности настройки КТ
PORTC = 0x00;  // Потушить диод, диод горит - КТ сформировалась корректно*/
 /*------Конец настройки КТ------*/
 PORTC |= (1<<PC6);  // РС6 - зажечь диод
 //UERST |= 0x01; // Сброс нулевой конечной точки
 //UERST = 0x00; // Очистка, для завершения броса и начала использования FIFO.
 UENUM = 0x00; // Выбор нулевой КТ

 UDCON  &= ~(1 << DETACH);

 while((UDINT & 0x04) != 0x04)
PORTC = 0x00;  // Потушить диод, ожидание флага Start of Frame
 PORTC |= (1<<PC6);  // РС6 - зажечь диод
 UDINT  &= ~(1 << SOFI);
 while((UEINTX & 0x08) != 0x08)
PORTC = 0x00;  // Потушить диод, ожидание пакета SETUP, бит RXSTPI
 PORTC |= (1<<PC6);  // РС6 - зажечь диод

 while(1);
 return 0;
}

Изменено пользователем IgorKossak
[codebox] для длинного кода, [code] - для короткого!!!

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

решил приступить к написанию программы для USB

Начинать надо не с прошивки, а со спецификации USB. Очень желательно также соотв. книжку Гука глянуть ("Шины PCI, USB и FireWire"). Без этого, наверное, ничего не выйдет. Ну и примеры готовые разберите, предлагаемые Атмелом для вашего МК. Писать все это самому - жизни не хватит ...

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Спасибо за совет. Читал и Гука, и Агурова, и перевод спецификации USB 1.1. Если не сложно можно дать прямую ссылку на пример Атмела.

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Спасибо за совет. Читал и Гука, и Агурова, и перевод спецификации USB 1.1. Если не сложно можно дать прямую ссылку на пример Атмела.

Любой соотв. апп. ноут для вашего МК у него на сайте берете и изучаете. Найти легко, например, AVR272:

 

http://www.atmel.com/products/microcontrol...x?tab=documents

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Остаётся вопрос, почему после того как ловится бит прихода SOF, бит прихода маркера SETUP не устанавливается?

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Остаётся вопрос, почему после того как ловится бит прихода SOF, бит прихода маркера SETUP не устанавливается?

Вот честно, если Вы будете упираться в аппаратный уровень потратите напрасно много времени. С USB все всегда неоднозначно. И много зависит от железа.

Вам правильно подсказали. Возьмите ГОТОВЫЙ пример HID устройства на USB для AT90USB162 и перелопатьте код под свои нужды.

В свое время быстро и незатейливо удалось реализовать CDC+HID на данной меге. И HID там простой и очень дружелюбный.

А если хотите все таки USB узнать "от сохи", тогда начинать надо с CY7C68013 и FX2 Development Guide. Вот там да, все очень корректно. Можно лазить с осциллографом по шине, ловить биты и узнать много, очень много абсолютно бесполезной информации. Если только Вы не собираетесь корку на USB писать руками :)

Изменено пользователем fractcon

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Вот честно, если Вы будете упираться в аппаратный уровень потратите напрасно много времени. С USB все всегда неоднозначно. И много зависит от железа.

Вам правильно подсказали. Возьмите ГОТОВЫЙ пример HID устройства на USB для AT90USB162 и перелопатьте код под свои нужды.

В свое время быстро и незатейливо удалось реализовать CDC+HID на данной меге. И HID там простой и очень дружелюбный.

А если хотите все таки USB узнать "от сохи", тогда начинать надо с CY7C68013 и FX2 Development Guide. Вот там да, все очень корректно. Можно лазить с осциллографом по шине, ловить биты и узнать много, очень много абсолютно бесполезной информации. Если только Вы не собираетесь корку на USB писать руками :)

Если не заруднит, не могли бы выложить программу созданного вами HID устройства на at90usb162. Заранее спасибо.

Изменено пользователем arbuz

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

День добрый. ...

После того как вы сделали atach, шина должна сбросить ваше устройство. Дождитесь EORSTI а не SOFI (while((UDINT & 0x08) != 0x08)), потом сконфигурируйте endpoint0 ещё раз. Шина может сбросить устройство ещё раз. Опять дождитесь EORSTI и снова конфигурируйте endpoint0. После этого уже можете дожидатся RXSTPI. Дальше согласно спецификации шины. Не забудте сразу после получения флага EORSTI его сбросить.

Еще прислушайтесь к советам kovigora, он мне здорово помог с USB.

Изменено пользователем IgorKossak
избыточное цитирование

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Я тут давным-давно выкладывал проект под 162 мегу: http://electronix.ru/forum/index.php?act=a...st&id=32315

Посмотрите, там все обрезано лишнее, остается минимум, требуемый для понимания работы.

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Спасибо за совет. Читал и Гука, и Агурова, и перевод спецификации USB 1.1. Если не сложно можно дать прямую ссылку на пример Атмела.

Вот тут http://www.fourwalledcubicle.com/index.php проект LUFA есть хорошие примеры реализации USB на AVR

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Моё предложение такое, что надо сначала проверить железо - всё ли в нём настраивается для работы с USB? Загрузить в контроллер стандартный пример USB-девайса - пусть комп хотя бы обнаружит, что "найдено новое устройство, USB-AVR клавиатура... или мышь... или флешка" или какие там примеры ATMEL выкладывает.

Если компьютер обнаружил некое устройство, и его видно в диспетчере устройств, с названием, с всякими VID/PID - тогда уже можно начинать писать программу самому. А то, если что-то не так с железом (ножка не пропаяна, кварц не той частоты и т.п.) - можно долго ждать SETUP-пакетов.

 

Если с железом всё в порядке, и настройки железа в норме, то вот последовательность энумерации:

1. Подключаем Pull_Up резистор к D+ (для FullSpeed, и к D- для LowSpeed).

2. Обнаружив Pull_Up резистор - Хост выставляет на шине состояние RESET (1й раз).

3. Хост выставляет состояние SUSPEND.

4. Хост снова выставляет RESET (2й раз).

5. И только после этого - приходит первый SETUP-пакет. В буфер придёт последовательность из 8 байт: 0x80, 0x06, 0x00, 0x01, 0x00, 0x00, 0x40, 0x00 - отправляем ACK.

6. Отправляем Хосту 8 байт ответа: 0x12, 0x01, 0x00, 0x02, 0x00, 0x00, 0x00, 0x08

7. Хост присылает пакет данных нулевой-длины (длина 0 !) - отправляем ACK.

8. Хост снова выставляет RESET на шину (3й раз)

9. Хост присылает SETUP-пакет присвоения адреса: 0x00, 0x05, 0xNN, 0x00, 0x00, 0x00, 0x00, 0x00 где NN присвоенный устройству адрес (1-127). Прописываем присвоенный адрес в соответствующий регистр контроллера, и отправляем ACK.

10. Отправляем Хосту пакет нулевой длины.

11.... дальше больше, но это потом, если хотя бы эти шаги работают ;)

 

Если всё в порядке, больше состояние RESET на шине появляться не будет. А если с обменом будут ошибки - то Хост ещё два раза повторит с пункта [4], после этого выставит команду SUSPEND, и затихнет до переподключения USB-устройства.

Изменено пользователем controller_m30

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Спасибо всем за подсказки, результат - положительный, пакет SETUP приходит и об этом свидельствует установка бита RXSTPI. Далее действия:

#define Usb_read_byte() (UEDATX)

  while((UEINTX & 0x08) != 0x08)
    PORTC = 0x00;  // Потушить диод, ожидание пакета SETUP, бит RXSTPI
  UEINTX &= ~(1 << RXSTPI); // Очистка RXSTPI, для подтверждения пакета setup
  PORTC |= (1<<PC6);  // РС6 - зажечь диод

  UEINTX &= ~(1 << RXOUTI); // Очистка бита RXOUTI

  bmRequestType = Usb_read_byte();  // тип запроса; bmRequestType - переменная типа unsigned char
  bmRequest      = Usb_read_byte();  // код запроса; bmRequest - переменная типа unsigned char

  if (bmRequestType == 0x80)
    PORTC |= (1<<PC6);  // РС6 - зажечь диод
  else PORTC = 0x00;     // Потушить диод

  if (bmRequest == 0x06)
    PORTC |= (1<<PC6);  // РС6 - зажечь диод
  else PORTC = 0x00;  // Потушить диод

Результат оказывается следующим, переменные bmRequestType и bmRequest содержат одно и тоже число 0х80. Вопрос, как необходимо считывать данные из UEDATX? ACK отправляется самостоятельно контроллером или необходимо это сделать программно? И правильный ли вобщем алгоритм действий?

Изменено пользователем arbuz

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

После того как поймали RXSTPI. сделайте 8 последовательных считываний из UEDATX и сохраните эти данные чтобы потом анализировать. Только после того как прочли 8 байт, сбросте RXSTPI. Это аппаратно иницирует ACK, программно ничего делать не нужно. Внимательнее прочтите даташит, вся информация оттуда.

Забыл спросить зачем вовремя SETUP-а устанавливаете бит STALLRQ ? Если не ошибаюсь, это делается когда вы от хоста получаете не поддерживаемую вашим устройством команду.

Аппаратура у вас вроде в порядке, так что изучайте даташит и 9-ый раздел спецификации.

Изменено пользователем vgo1

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

После того как поймали RXSTPI. сделайте 8 последовательных считываний из UEDATX и сохраните эти данные чтобы потом анализировать. Только после того как прочли 8 байт, сбросте RXSTPI. Это аппаратно иницирует ACK, программно ничего делать не нужно. Внимательнее прочтите даташит, вся информация оттуда.

Забыл спросить зачем вовремя SETUP-а устанавливаете бит STALLRQ ? Если не ошибаюсь, это делается когда вы от хоста получаете не поддерживаемую вашим устройством команду.

Аппаратура у вас вроде в порядке, так что изучайте даташит и 9-ый раздел спецификации.

Спасибо за подробное разъяснение, бит STALLRQ устанавливался в первоначальной версии прошивки, на данном этапе установка бита отсутствует.

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Процесс пошёл, пришёл пакет SETUP, флаг RXSTPI установился, далее считываю данные из UEDATX, и далее флаг RXSTPI сбрасываю.

  while((UEINTX & 0x08) != 0x08)
    PORTC = 0x00;  // Потушить диод, ожидание пакета SETUP, бит RXSTPI
  PORTC |= (1<<PC6);  // РС6 - зажечь диод

  bmRequestType = Usb_read_byte();  // тип запроса
  bmRequest     = Usb_read_byte();    // код запроса

  if (bmRequestType == 0x80)
    PORTC |= (1<<PC6);  // РС6 - зажечь диод
  else PORTC = 0x00;  // Потушить диод

  if (bmRequest == 0x06)
    PORTC |= (1<<PC6);  // РС6 - зажечь диод
  else PORTC = 0x00;  // Потушить диод*/


  UEINTX &= ~(1 << RXSTPI); // Очистка RXSTPI, для отправки ACK

 

Далее, должен прийти пакет IN, в ответ на который контроллер обязан выслать 8 байт дескриптора, и получить от хоста ACK.

#define Usb_write_byte(byte)  (UEDATX = (unsigned char)byte)

 while((UEINTX & 0x01) != 0x01)  //(ВЕРНО?)
   PORTC = 0x00;  // Потушить диод, ожидание пакета IN, бит TXINI 
 PORTC |= (1<<PC6);  // РС6 - зажечь диод 
 Usb_write_byte(0x12);
 Usb_write_byte(0x01);
 Usb_write_byte(0x00);
 Usb_write_byte(0x02);
 Usb_write_byte(0x00);
 Usb_write_byte(0x00);
 Usb_write_byte(0x00);
 Usb_write_byte(0x08);  // Запись данных в UEDATX

 UEINTX &= ~(1 << TXINI); // Сброс TXINI для отправки данных хосту

 

Вопрос: правильно ли, то что установка бита TXINI считается приходом пакета IN? Каким образом узнать что хост прислал ACK?

 

Далее, хост присылает пакет OUT нулевой длинны, задача контроллера отправить пакет ACK.

  while((UEINTX & 0x04) != 0x04)
    PORTC = 0x00;  // Потушить диод, ожидание пакета OUT, бит RXOUTI
  PORTC |= (1<<PC6);  // РС6 - зажечь диод
  
  UEINTX = 0x00; // Сброс регистра, для отправки ACK

 

Вопрос: для отправки ACK на пакет OUT достаточно очистки бита RXOUTI?

Изменено пользователем IgorKossak
Поднятие темы запрещено правилами форума!!!

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.

Гость
Ответить в этой теме...

×   Вставлено с форматированием.   Вставить как обычный текст

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

×
×
  • Создать...