新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > 如何使用STM32的USB非控制端點發送多個數據包

        如何使用STM32的USB非控制端點發送多個數據包

        作者: 時間:2016-12-02 來源:網絡 收藏
        以下是網友提出的問題和我對這個問題的說明。
        SMT32F103,根據例程Custom_HID修改,利用EP1 以EP_INTERRUPT 的方式發送包,原來的例程每次發送2個字節,現在修改后包的長度不超過64字節時發送是正常的,但當一個包長超過64字節時就發送失敗,沒有數據出來(程序沒有死機),該改的地方都已經修改了,不知道哪個地方還沒有改到位,謝謝!
        現象就是 超過63字節的包死活也發不出去,而且發送包的大小 還與 CustomHID_ConfigDescriptor里面的 EP1 IN endpoint 描述里包大小有關 ,沒道理啊,其他的MCU 這地方設置為8 照樣發送256B 以上的包。
        在Custom_HID例程上修改了如下代碼:
        1.usb_proc.c 的CustomHID_Reset()里 SetEPTxCount(ENDP1, 64);
        2.關閉 DMA中斷,不讓ADC采樣后發送EP1包
        3.在main.c里 重復發送一個128B的包,
        while(1){
        for(i=0;i<2;i++)
        { SetEPTxAddr(ENDP1, ENDP1_TXADDR+i*64);
        SetEPTxValid(ENDP1);
        Delay(10000);
        }
        }
        4. 由于一個包是128B,最大包長是64B,所以分兩次發送出來,奇怪的是所有例程發送包時都沒有查發送狀態的處理,也沒有找到相應的狀態等待函數,這樣的話,是不是出現第一個包還沒有發送完,第二個包就沖掉了第一個包的數據?
        5. 所以問題很簡單,就是如何發送一個多數據包,發送函數要如何寫?
        以下是關于這個問題的解答:
        分兩次發送是對的,但關鍵是每次發送前需要檢查上次發送是否完成。
        檢查一個端點的發送是否結束有2種方法,第一種方法是當發送結束(設備收到ACK)時,有一個發送結束中斷,這個中斷由USB庫處理,并通過EP1_IN_Callback這個回調函數交由用戶程序確認,你可以搜索一下,例子中把EP1_IN_Callback定義為NOP_Process,沒有處理這個回調事件。如果要用這種方法檢測端點發送結束,你需要自己定義回調函數并做相應處理。
        檢測端點發送結束的另一個方法是查詢這個端點的狀態,如果端點狀態處于EP_TX_VALID,說明發送未結束,如果端點狀態處于EP_TX_NAK,說明發送結束。使用下述調用可以得到端點1的發送狀態:
        GetEPTxStatus(ENDP1)
        按照你的思路,可以使用第二種方法實現發送多個數據包的功能。
        假定要發送150個字節的MyBuffer,EP1的最大包長設為64字節。
        u8 MyBuffer[150];
        int packetN;
        packetN = 3;
        while (1) {
        if (packetN < 3) { // 有數據需要發送時置packetN為0
        if (GetEPTxStatus(ENDP1) == EP_TX_NAK) {
        if (packetN == 0) { // 拷貝頭64字節到發送緩沖區
        UserToPMABufferCopy(MyBuffer, ENDP1_TXADDR, 64);
        SetEPTxCount(ENDP1, 64);
        }
        else if (packetN == 1) { // 拷貝第2個64字節到發送緩沖區
        UserToPMABufferCopy(MyBuffer+64, ENDP1_TXADDR, 64);
        SetEPTxCount(ENDP1, 64);
        }
        else if (packetN == 2) { // 拷貝最后22字節到發送緩沖區
        UserToPMABufferCopy(MyBuffer+128, ENDP1_TXADDR, 22);
        SetEPTxCount(ENDP1, 22);
        }
        packetN++;
        SetEPTxStatus(ENDP1, EP_TX_VALID);
        }
        }
        ...... // 其它操作
        }
        這里使用了一個變量記錄應該發送第幾個數據包,當程序的其它部分準備好數據后只要設置這個變量packetN=0,上述發送操作就會啟動,程序的其它部分只需檢測packetN==3即可知道MyBuffer是否已經騰空,程序的其它部分可以使用MyBuffer繼續其它操作,注意這時數據不一定已經全部發送完畢。
        你的另一個問題在于這一行:SetEPTxAddr(ENDP1, ENDP1_TXADDR+i*64);
        ENDP1_TXADDR是專門的發送緩沖區,它的長度是有限的,而且是每32位編址中只有低16位有效;所以需要使用函數UserToPMABufferCopy()操作這個發送緩沖區,這個函數已經在USB庫的手冊中說明。
        最后一個問題是:如果你的程序中使用了ENDP1_RXADDR,因為你改變了ENDP1包的長度,即改變了發送緩沖區的長度,需要在usb_conf.h中重新定義以下ENDP1_RXADDR的地址。


        評論


        技術專區

        關閉
        主站蜘蛛池模板: 清涧县| 铁岭市| 明光市| 正镶白旗| 新源县| 榆中县| 册亨县| 浦江县| 宁乡县| 赫章县| 凯里市| 宕昌县| 寻甸| 贡觉县| 大同县| 正镶白旗| 扶绥县| 涪陵区| 乐昌市| 怀远县| 新巴尔虎右旗| 长沙县| 额尔古纳市| 日照市| 桦南县| 全椒县| 安塞县| 蛟河市| 潞西市| 丽水市| 巴青县| 赣榆县| 合山市| 德令哈市| 大足县| 富民县| 桑日县| 沙湾县| 平利县| 思茅市| 大同市|