STM32 USB主机传输中断过程

本文详细介绍了STM32FUSBOTGFS作为主机时的中断过程和OUT、IN传输时序。在OUT传输中断中,应用程序通过usb主机驱动发送数据,经过分包并触发一系列中断完成数据传输。而在IN传输中断中,主机从设备接收数据,通过不断发出IN令牌直至设备响应。整个过程涉及到USB协议栈、中断处理和底层驱动交互。
摘要由CSDN通过智能技术生成

1、STM32F USB主机


       本文介绍STM32F USB OTG FS主机的中断过程及时序,关于USB通信的原理与报文不在本文的介绍,请参考其他文档。本文根据实际的使用USB主机详细的描述的USB主机中断发生的过程及顺序,是全网少有的文章。本文介绍所涉及的USB主机程序请参考rt thread操作系统的USB主机协议栈与USB主机驱动。

2、USB主机OUT传输中断     

       应用程序调用USB主机驱动发送数据到USB设备,这个过程会在USB总线上面产生OUT传输过程。即应用程序调用usb主机rt_usb_hcd_pipe_xfer函数,此函数会对发送的数据长度按端点传输包的大小进行分包,每次发送一个分包,发送多个分包后才完成本次的数据传输。rt_usb_hcd_pipe_xfer最后调用usb底层驱动函数drv_pipe_xfer,调用HAL_HCD_HC_SubmitRequest函数来向USB主机发出一个OUT的URB请求,让usb主机在usb总线上输出一个数据分包。usb主机会在usb总线上发出一个OUT令牌,再发出一个OUT数据包,USB设备成功接收后,发出ACK给主机此时主机产生OUT ACK中断,再产生OUT XFRC中断(传输完成),最后再产生OUT CHH中断(通信停止),此时向驱动程序drv_pipe_xfer发送传输完成信号量,通知驱动完成此次分包传输。当发送的数据长度大于一个端点数据包长度时,会再进行多次如上的操作。具体过程如下图所示。

int rt_usb_hcd_pipe_xfer(uhcd_t hcd, upipe_t pipe, void* buffer, int nbytes, int timeout)
{
    rt_size_t remain_size;
    rt_size_t send_size;
    remain_size = nbytes;
    int  len = 0;

    if(pipe->ep.bEndpointAddress & 0x80)
    {
        /*control ep*/
        len = hcd->ops->pipe_xfer(pipe, USBH_PID_DATA, buffer, nbytes, timeout);
        if((len >=0) && (len <= nbytes))
        {
            
            return len;
        }
        else
        {
            return 0;
        }
    }
    else
    {
        /*OUT 传输在这里进行了分包*/
        rt_uint8_t * pbuffer = (rt_uint8_t *)buffer;
        do
        {
            RT_DEBUG_LOG(RT_DEBUG_USB,("pipe transform remain size,: %d\n", remain_size));
            send_size = (remain_size > pipe->ep.wMaxPacketSize) ? pipe->ep.wMaxPacketSize : remain_size;
            if(hcd->ops->pipe_xfer(pipe, USBH_PID_DATA, pbuffer, send_size, timeout) == send_size)
            {
                remain_size -= send_size;
                pbuffer += send_size;
            }
            else
            {
                return 0;
            }
        }while(remain_size > 0);

        return nbytes;
    }
    
}

3、USB主机IN传输中断 

      USB主机要从设备中接收一段数据时,应用程序会调用rt_usb_hcd_pipe_xfer函数,此函数把接收数据的缓冲区地址和长度传输给底层驱动drv_pipe_xfer函数,清除信号量,调用HAL_HCD_HC_SubmitRequest函数来向USB主机发出一个IN的URB请求。USB主机在USB总线上发出IN令牌,当usb设备有无数据传输时返回NAK,并产生USB主机IN NAK中断;当USB设备有数据传输时,发生IN RXQLVL中断(接收水平),USB设备根据要传输的数据总长按端点最大数据包长度进行分包发送,发生IN RXQLVL的次数就是分包的数量,所以说主机IN传输的分包是由USB控制器来分包的,与OUT传输不一样。USB设备发送数据 完成后,USB主机发出IN ACK中断,再发生IN XFRC中断,IN CHH中断,此时发送信号量通知驱动程序已经接收到数据。

       如果应用程序想一直等待设备发送出数据,驱动程序中等待接收的信号量超时时间无线长就可以了,USB主机会在总线完全空闲时不断的发出IN令牌,直到USB设备有数据发送后停止此次IN传输。

 

 

  • 3
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
STM32USB中断主要有两种类型:USB中断USB中断EP中断。 1. USB中断USB中断是指在USB传输过程中出现错误或完成传输时触发的中断。例如,当USB主机发送数据给USB设备时,如果USB设备接收到数据,则会触发USB中断。在STM32中,USB中断具有以下优先级:USB中断>系统中断,因此当USB中断和系统中断同时触发时,USB中断会先被处理。 2. USB中断EP中断USB中断EP中断是指在USB传输过程中,当每个端点完成数据收发时触发的中断。例如,当USB主机发送数据给USB设备的某个端点时,如果该端点接收到数据,则会触发USB中断EP中断。在STM32中,USB中断EP中断具有以下优先级:USB中断>USB中断EP中断>系统中断,因此当USB中断USB中断EP中断和系统中断同时触发时,USB中断优先级最高,其次是USB中断EP中断,最后是系统中断。 在使用STM32USB中断时,需要根据不同的中断类型选择不同的中断处理函数,并配置相应的中断优先级。例如,在使用USB中断EP中断时,可以使用以下函数进行中断处理: void USB_LP_CAN1_RX0_IRQHandler(void) 其中,USB_LP_CAN1_RX0_IRQHandler是STM32USB中断EP中断的处理函数。在处理中断时,需要先判断是哪个端点的中断,并根据不同的端点进行相应的操作。例如,如果是端点1的中断,则可以使用以下代码进行处理: if(USB_GetITStatus(ENDP1, USB_IT_RXFE) == SET) { //处理端点1的接收中断 //... } 以上是STM32USB中断的基本知识,希望能对你有所帮助。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值