STM32F7 Discovery - обнаружение режима хоста / устройства USB FS

Плата STM32F723IEK6 Discovery имеет полноскоростной интерфейс USB. Я тщетно пытаюсь его инициализировать. Сигнал сброса никогда не принимается от хоста, и соответствующий флаг прерывания не установлен.

Интерфейс FS - OTG. Предполагается определить напряжение VBUS и состояние контакта ID, чтобы определить, подключен ли он как хост или как устройство. В режиме устройства хост должен обеспечивать VBUS, а контакт ID должен быть отсоединен и вынут ВЫСОКОМ. Когда устройство обнаруживает соединение, оно должно потянуть за контакт DP, чтобы указать соединение с хостом. Затем хост отправляет сигнал RESET, опуская линии данных на низкий уровень. Это теория.

Похоже, что плата не тянет высоко линию DP. В конфигурации OTG по умолчанию (как в программе ниже) с включенным обнаружением VBUS в регистре GINTSTS всегда устанавливаются только биты CIDSCHG (изменение идентификатора), SRQINT (сеанс) и SOF. Бит CMOD равен нулю, указывая на режим устройства. Идентификационная линия кажется низкой, даже если кабель не подключен.

Я настраиваю соответствующие контакты (A9, A10, A11, A12) в режиме AF 10 (OTG FS). Однако это может быть ненужным, поскольку FS PHY, кажется, подключен непосредственно к контактам, минуя мультиплексор GPIO (не уверен).

Я попытался форсировать режим устройства и отключить обнаружение VBUS, но это не вызвало никакого эффекта, а также вытащил линию ID из GPIO.

У меня тот же код работает на плате STM32F4 раньше и не было таких проблем.

Я хотел бы понять, почему не работает обнаружение соединения.

Код ниже, извините за многословность, он имеет часы, GPIO и код инициализации USB, некоторые вспомогательные функции опущены.

Справочное руководство по STM32F72x (15 МБ!) http://www.st.com/resource/en/reference_manual/dm00305990.pdf

Лист данных http://www.st.com/resource/en/datasheet/DM00330506.pdf

Руководство по Правлению http://www.st.com/resource/en/user_manual/dm00342318.pdf

#include "stm32f7xx.h"

#define PLL_M 25
#define PLL_N 336
#define PLL_P 0
#define PLL_Q 7
#define SYS_FREQ 168000000

void rcc_config(void)
{
    RCC->APB1ENR |= RCC_APB1ENR_PWREN;
    PWR->CR1 = (PWR->CR1 & ~PWR_CR1_VOS_Msk)
            | PWR_CR1_VOS_1;
    RCC->CR |= RCC_CR_HSEON;
    while ((RCC->CR & RCC_CR_HSERDY) == 0);
    RCC->PLLCFGR = PLL_M | (PLL_N << 6) | (PLL_P << 16) | RCC_PLLCFGR_PLLSRC_HSE | (PLL_Q << 24);
    RCC->CR |= RCC_CR_PLLON;
    RCC->CFGR = (RCC->CFGR & ~(RCC_CFGR_HPRE_Msk | RCC_CFGR_PPRE1_Msk | RCC_CFGR_PPRE2_Msk))
            | RCC_CFGR_HPRE_DIV1
            | RCC_CFGR_PPRE2_DIV2
            | RCC_CFGR_PPRE1_DIV4;
    FLASH->ACR = FLASH_ACR_PRFTEN | FLASH_ACR_ARTEN | FLASH_ACR_LATENCY_5WS;
    while ((RCC->CR & RCC_CR_PLLRDY) == 0);
    while ((PWR->CSR1 & PWR_CSR1_VOSRDY) == 0);
    RCC->CFGR &= RCC_CFGR_SW;
    RCC->CFGR |= RCC_CFGR_SW_PLL;
    while ((RCC->CFGR & RCC_CFGR_SWS) != RCC_CFGR_SWS_PLL);
}

#define GPIO_OTYPE_PP 0
#define GPIO_OTYPE_OD 1
#define GPIO_PULLUP   1

void gpio_config_mode(GPIO_TypeDef* gpio, unsigned pin, unsigned mode)
{
    gpio->MODER = (gpio->MODER & ~(3u << (2 * pin))) | (mode << (2 * pin));
}
void gpio_config_in(GPIO_TypeDef* gpio, unsigned pin)
{
    gpio_config_mode(gpio, pin, 0);
}
void gpio_config_out(GPIO_TypeDef* gpio, unsigned pin, unsigned otype, unsigned ospeed)
{
    gpio_config_mode(gpio, pin, 1);
    gpio->OTYPER  = (gpio->OTYPER  & ~(1u << (1 * pin))) | (otype  << (1 * pin));
    gpio->OSPEEDR = (gpio->OSPEEDR & ~(3u << (2 * pin))) | (ospeed << (2 * pin));
}
void gpio_config_af(GPIO_TypeDef* gpio, unsigned pin, unsigned af)
{
    gpio_config_mode(gpio, pin, 2);
    unsigned pin_group = pin >> 3;
    unsigned pin_offset = pin & 7;
    gpio->AFR[pin_group] = (gpio->AFR[pin_group] & ~(0xf << (pin_offset * 4)))
            | (af << (pin_offset * 4));
}
void gpio_config_pullup(GPIO_TypeDef* gpio, unsigned pin, unsigned pupd)
{
    gpio->PUPDR = (gpio->PUPDR   & ~(3u << (2 * pin))) | (pupd   << (2 * pin));
}

USB_OTG_GlobalTypeDef *usb = USB_OTG_FS;
USB_OTG_DeviceTypeDef *usb_dev = (USB_OTG_DeviceTypeDef *)(USB_OTG_FS_PERIPH_BASE + USB_OTG_DEVICE_BASE);

void usb_config(void)
{
    /* The application must program this register before starting any transactions
     * on either the AHB or the USB. Do not make changes to this register after
     * the initial programming. */
    usb->GUSBCFG |= USB_OTG_GUSBCFG_PHYSEL; // TODO: no effect for F7, read-only bit
//  usb->GUSBCFG |= USB_OTG_GUSBCFG_FDMOD;
    /* After setting the force bit, the application must wait at least * 25 ms
     * before the change takes effect. */
    delay_ms(25);

    // USB core reset
    while ((usb->GRSTCTL & USB_OTG_GRSTCTL_AHBIDL) == 0);
    usb->GRSTCTL |= USB_OTG_GRSTCTL_CSRST;
    while ((usb->GRSTCTL & USB_OTG_GRSTCTL_CSRST) != 0);
    delay_us(1); // actually, 3 PHY clocks

    usb_dev->DCFG = USB_OTG_DCFG_DSPD_0 | USB_OTG_DCFG_DSPD_1; // full speed
//  usb->GAHBCFG = 0;
//  usb->PCGCTL = 0;
    usb->GCCFG |= USB_OTG_GCCFG_VBDEN; // VBUS detection
    usb->GCCFG |= USB_OTG_GCCFG_PWRDWN; // enable PHY

/// usb->GINTSTS= 0xFFFFFFFF;
//  usb->GINTMSK = 0;
//  usb->GINTSTS = 0xFFFFFFFF;
//  usb->GAHBCFG |= USB_OTG_GAHBCFG_GINT;
}

void usb_poll(void)
{
    uint32_t intsts = usb->GINTSTS;
    if (intsts & USB_OTG_GINTSTS_USBRST)
        usb->GINTSTS = USB_OTG_GINTSTS_USBRST;
    if (intsts & USB_OTG_GINTSTS_RSTDET)
        usb->GINTSTS = USB_OTG_GINTSTS_RSTDET;
    if (intsts & USB_OTG_GINTSTS_ENUMDNE)
        usb->GINTSTS = USB_OTG_GINTSTS_ENUMDNE;
    if (intsts & USB_OTG_GINTSTS_CIDSCHG)
        usb->GINTSTS = USB_OTG_GINTSTS_CIDSCHG;
    if (intsts & USB_OTG_GINTSTS_SRQINT)
        usb->GINTSTS = USB_OTG_GINTSTS_SRQINT;
}

#define LED_PIN 5
#define USB_AF 10

int main()
{
    rcc_config();

    SysTick->LOAD = 0xffffffu;
    SysTick->VAL = 0;
    SysTick->CTRL = 5;

    RCC->AHB1ENR |= RCC_AHB1ENR_GPIOAEN;
    gpio_config_out(GPIOA, LED_PIN, GPIO_OTYPE_PP, 0);
    gpio_config_af(GPIOA,  9, USB_AF); // VBUS_DET
    gpio_config_af(GPIOA, 10, USB_AF); // ID
    gpio_config_af(GPIOA, 11, USB_AF); // DM
    gpio_config_af(GPIOA, 12, USB_AF); // DP

    RCC->AHB2ENR |= RCC_AHB2ENR_OTGFSEN;
    usb_config();
    while (1)
        usb_poll();
}

1 ответ

Решение

Конечно.

Регистр OTG_FS_DCTL:

Бит 1

SDIS:

Мягкое отключение Приложение использует этот бит для подачи сигнала ядру USB OTG на мягкое отключение. Пока этот бит установлен, хост не видит, что устройство подключено, и устройство не получает сигналы по USB. Ядро остается в отключенном состоянии, пока приложение не очистит этот бит. 0: Нормальная работа. Когда этот бит очищается после программного отключения, ядро ​​генерирует событие подключения устройства к USB-хосту. Когда устройство повторно подключено, хост USB перезапускает перечисление устройства. 1: ядро ​​генерирует событие отключения устройства от хоста USB.

Из руководства STM32F411:

Сбросить значение: 0x0000 0000

Из руководства STM32F723:

Сбросить значение: 0x0000 0002

До тех пор, пока приложение не изменит значение сброса, USB-блок STM32F7 запускается в состоянии "мягкого отключения", а STM32F4 пытается подключиться сразу после обнаружения VBUS. В документации (например, раздел 32.6.2 Периферийные состояния) этот факт вообще не упоминается.

В моем случае очистка бита SDIS включила успешное установление соединения.

Другие вопросы по тегам