There's a nasty bug hidden in USBD_HS_STM32F4xx.c driver code. It ALWAYS enables on-chip HSUSB PHY even if device is configured to use external PHY, making PB15 wrongly routed to PHY instead of another configured peripheral (SPI2 MOSI in my case).
Fix is very simple:
static int32_t USBD_DeviceConnect (void) { if (!(otg_hs_state & OTG_HS_USBD_DRIVER_POWERED) ) { return ARM_DRIVER_ERROR; } if ( otg_hs_state & OTG_HS_USBD_DRIVER_CONNECTED) { return ARM_DRIVER_OK; } OTG->DCTL &= ~OTG_HS_DCTL_SDIS; // Soft disconnect disabled #ifndef RTE_USB_OTG_HS_PHY OTG->GCCFG |= OTG_HS_GCCFG_PWRDWN; #endif otg_hs_state |= OTG_HS_USBD_DRIVER_CONNECTED; return ARM_DRIVER_OK; }
It caused me a lot of headache, so I would like to share it with anyone that might have the similar problem.