www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁 > > 嵌入式案例Show
[導(dǎo)讀]本篇主要介紹U盤模塊,必須是有USB-HOST功能的芯片才可以進(jìn)行U盤數(shù)據(jù)的讀寫,在硬件設(shè)計時已經(jīng)提到了,本例用到的芯片屬于STM32F105系列是帶有OTG功能的,103系列的芯片只能作為從設(shè)備。

本篇主要介紹U盤模塊,必須是有USB-HOST功能的芯片才可以進(jìn)行U盤數(shù)據(jù)的讀寫,在硬件設(shè)計時已經(jīng)提到了,本例用到的芯片屬于STM32F105系列是帶有OTG功能的,103系列的芯片只能作為從設(shè)備。STM32F105 USB接口主要特點:

USB2.0協(xié)議,OTG1.3協(xié)議。

可作為USB主機(jī)、USB設(shè)備、

OTG設(shè)備(A類/B類)使用

可使用內(nèi)部FS PHY做FS通信

從官方下載USB例程,Library庫如下圖

我們需要做U盤數(shù)據(jù)的讀寫,USB做主機(jī)即HOST功能。如上圖在官方例程的Libraries目錄中。其中的USB OTG是USB Device和USB Host的基礎(chǔ)。在實際使用中,USB OTG是USB Device和USB Host的底層驅(qū)動。本例需要移植OTG+HOST。(現(xiàn)在可以不需要移植了,通過STM32CubeMX直接由圖形界面簡單配置下,生成初始化代碼,并對外設(shè)做了進(jìn)一步的抽象,讓開發(fā)人員更只專注應(yīng)用的開發(fā)。搞嵌入式還是要了解底層驅(qū)動,對吧?)

01

驅(qū)動文件移植

移植文件如下圖:

1.1 OTG和HOST驅(qū)動文件簡要說明

本例中需要實現(xiàn)U盤讀寫需要MSC驅(qū)動程序,MSC:(Mass storage class)大容量存儲類驅(qū)動程序用于支持通用USB閃存驅(qū)動程序,使用BOT“Bulk-Only Transport”協(xié)議和透明SCSI命令集。

1.2 MSC驅(qū)動文件介紹

1.3 用戶文件介紹

通常所有的用戶代碼均在以上文件做修改即可,只有少數(shù)特殊處理會設(shè)計到驅(qū)動程序源碼的修改。在配置好后,一般來說,所有修改均在usbh_usr.h/.c中。

1.4 配置文件介紹

usb_conf.h

#ifndef USE_USB_OTG_FS //#define USE_USB_OTG_FS#endif /* USE_USB_OTG_FS */ #ifdef USE_USB_OTG_FS  #define USB_OTG_FS_CORE //使能內(nèi)核的全速模式 #endif #ifndef USE_USB_OTG_HS //#define USE_USB_OTG_HS #endif /* USE_USB_OTG_HS */ #ifndef USE_ULPI_PHY //#define USE_ULPI_PHY#endif /* USE_ULPI_PHY */ #ifndef USE_EMBEDDED_PHY //#define USE_EMBEDDED_PHY#endif /* USE_EMBEDDED_PHY */ #ifdef USE_USB_OTG_HS  #define USB_OTG_HS_CORE //使能內(nèi)核的高速模式#endif/****************** USB OTG HS CONFIGURATION **********************************/#ifdef USB_OTG_HS_CORE #define RX_FIFO_HS_SIZE 512 //設(shè)置高速模式下接收的FIFO的大小 #define TXH_NP_HS_FIFOSIZ 256 //設(shè)置高速模式下,作為USB Host時,非周期性發(fā)送的FIFO的大小 #define TXH_P_HS_FIFOSIZ 256 //設(shè)置高速模式下,作為USB Host時,周期性發(fā)送的FIFO的大小  #ifdef USE_ULPI_PHY #define USB_OTG_ULPI_PHY_ENABLED #endif #ifdef USE_EMBEDDED_PHY #define USB_OTG_EMBEDDED_PHY_ENABLED #endif #define USB_OTG_HS_INTERNAL_DMA_ENABLED #define USB_OTG_EXTERNAL_VBUS_ENABLED// #define USB_OTG_INTERNAL_VBUS_ENABLED#endif/****************** USB OTG FS CONFIGURATION **********************************/#ifdef USB_OTG_FS_CORE #define RX_FIFO_FS_SIZE 128 //設(shè)置全速模式下接收的FIFO的大小 #define TXH_NP_FS_FIFOSIZ 96 //設(shè)置全速模式下,作為USB Host時,非周期性發(fā)送的FIFO的大小 #define TXH_P_FS_FIFOSIZ 96 //設(shè)置全速模式下,作為USB Host時,周期性發(fā)送的FIFO的大小#endif/****************** USB OTG MODE CONFIGURATION ********************************/#define USE_HOST_MODE //HOST 模式//#define USE_DEVICE_MODE//#define USE_OTG_MODE 

usbh_config.h

#define USBH_MAX_NUM_ENDPOINTS 2 //支持端點的最大數(shù)量 1 bulk IN + 1 bulk Out #define USBH_MAX_NUM_INTERFACES 2 //支持接口的最大數(shù)量#ifdef USE_USB_OTG_FS #define USBH_MSC_MPS_SIZE 0x40#else#define USBH_MSC_MPS_SIZE 0x200#endif 


02

USB處理狀態(tài)機(jī)

下面介紹USB處理的核心狀態(tài)機(jī)

核心狀態(tài)機(jī)過程由USBH_Process函數(shù)實現(xiàn)。應(yīng)該從應(yīng)用程序主循環(huán)周期性地調(diào)用該函數(shù)。

/*** @brief USBH_Process* USB Host core main state machine process* @param None * @retval None*/void USBH_Process(USB_OTG_CORE_HANDLE *pdev , USBH_HOST *phost){ volatile USBH_Status status = USBH_FAIL;   /* check for Host port events */ if ((HCD_IsDeviceConnected(pdev) == 0)&& (phost->gState != HOST_IDLE))  { if(phost->gState != HOST_DEV_DISCONNECTED)  { phost->gState = HOST_DEV_DISCONNECTED; } }  switch (phost->gState) {  case HOST_IDLE ://在主機(jī)初始化之后,內(nèi)核在這種狀態(tài)下開始輪詢USB設(shè)備連接。 當(dāng)檢測到設(shè)備斷開連接事件時以及未發(fā)生錯誤時也會進(jìn)入此狀態(tài)。  if (HCD_IsDeviceConnected(pdev))  { phost->gState = HOST_DEV_ATTACHED; USB_OTG_BSP_mDelay(100); } break;  case HOST_DEV_ATTACHED ://當(dāng)一個設(shè)備連接時,核心進(jìn)入這個狀態(tài)。 當(dāng)一個設(shè)備被檢測到時,狀態(tài)機(jī)轉(zhuǎn)到HOST_ENUMERATION狀態(tài)。  phost->usr_cb->DeviceAttached(); phost->Control.hc_num_out = USBH_Alloc_Channel(pdev, 0x00); phost->Control.hc_num_in = USBH_Alloc_Channel(pdev, 0x80);   /* Reset USB Device */ if ( HCD_ResetPort(pdev) == 0) { phost->usr_cb->ResetDevice(); /* Wait for USB USBH_ISR_PrtEnDisableChange()  Host is Now ready to start the Enumeration  */  phost->device_prop.speed = HCD_GetCurrentSpeed(pdev);  phost->gState = HOST_ENUMERATION; phost->usr_cb->DeviceSpeedDetected(phost->device_prop.speed);  /* Open Control pipes */ USBH_Open_Channel (pdev, phost->Control.hc_num_in, phost->device_prop.address, phost->device_prop.speed, EP_TYPE_CTRL, phost->Control.ep0size);   /* Open Control pipes */ USBH_Open_Channel (pdev, phost->Control.hc_num_out, phost->device_prop.address, phost->device_prop.speed, EP_TYPE_CTRL, phost->Control.ep0size);  } break;  case HOST_ENUMERATION: //在這種狀態(tài)下,核心進(jìn)行USB設(shè)備的基本枚舉。 在枚舉過程結(jié)束時,選擇默認(rèn)設(shè)備配置(配置0)。 /* Check for enumeration status */  if ( USBH_HandleEnum(pdev , phost) == USBH_OK) {  /* The function shall return USBH_OK when full enumeration is complete */  /* user callback for end of device basic enumeration */ phost->usr_cb->EnumerationDone();  phost->gState = HOST_USR_INPUT;  } break;  case HOST_USR_INPUT: //這是一個中間狀態(tài),它遵循枚舉并包括等待用戶輸入以啟動USB類操作。 /*The function should return user response true to move to class state */ if ( phost->usr_cb->UserInput() == USBH_USR_RESP_OK) { if((phost->class_cb->Init(pdev, phost))\ == USBH_OK) { phost->gState = HOST_CLASS_REQUEST;  }  }  break;  case HOST_CLASS_REQUEST: //從此狀態(tài)開始,類驅(qū)動程序接管,并調(diào)用類請求狀態(tài)機(jī)以處理所有初始類控制請求。 完成所需的類請求后,內(nèi)核將移至HOST_CLASS狀態(tài)。 /* process class standard contol requests state machine */  status = phost->class_cb->Requests(pdev, phost);  if(status == USBH_OK) { phost->gState = HOST_CLASS; }   else { USBH_ErrorHandle(phost, status); }   break;  case HOST_CLASS: // 在這種狀態(tài)下,類狀態(tài)機(jī)被稱為類相關(guān)操作(非控制和控制操作) /* process class state machine */ status = phost->class_cb->Machine(pdev, phost); USBH_ErrorHandle(phost, status); break;   case HOST_CTRL_XFER: //每當(dāng)需要控制轉(zhuǎn)移時就會進(jìn)入該狀態(tài) /* process control transfer state machine */ USBH_HandleControl(pdev, phost);  break;  case HOST_SUSPENDED: break;  case HOST_ERROR_STATE: //只要有任何庫狀態(tài)機(jī)發(fā)生未恢復(fù)的錯誤,就會進(jìn)入此狀態(tài)。 在這種情況下,調(diào)用用戶回調(diào)函數(shù)(例如,顯示未恢復(fù)的錯誤消息)。 然后主機(jī)庫被重新初始化。 /* Re-Initilaize Host for new Enumeration */ USBH_DeInit(pdev, phost); phost->usr_cb->DeInit(); phost->class_cb->DeInit(pdev, &phost->device_prop); break;  case HOST_DEV_DISCONNECTED ://設(shè)備斷開連接進(jìn)入此狀態(tài)  /* Manage User disconnect operations*/ phost->usr_cb->DeviceDisconnected();  /* Re-Initilaize Host for new Enumeration */ USBH_DeInit(pdev, phost); phost->usr_cb->DeInit(); phost->class_cb->DeInit(pdev, &phost->device_prop);  USBH_DeAllocate_AllChannel(pdev);  phost->gState = HOST_IDLE;  break;  default : break; }} 

MSC核心狀態(tài)機(jī),U盤讀寫操作的上層實現(xiàn),代碼如下:

int USBH_USR_MSC_Application(void){ uint16_t bytesWritten, bytesToWrite; FILINFO fno; DIR dir; /**********狀態(tài)機(jī)******************/ switch(USBH_USR_ApplicationState) { case USH_USR_FS_INIT:  USBH_USR_ApplicationState = USH_USR_IDLE;  /* Initialises the File System*/ if (f_mount( 0, &fatfs ) != FR_OK ) //掛載失敗 { return(-1); } if(USBH_MSC_Param.MSWriteProtect == DISK_WRITE_PROTECTED)//寫保護(hù) { return(-1); } if(f_open(&file,"TEST.txt", FA_WRITE | FA_CREATE_ALWAYS) == FR_OK)//U盤有效 {  bytesToWrite = sizeof(writeTextBuff);  res= f_write (&file, writeTextBuff, bytesToWrite, (void *)&bytesWritten); //寫一個測試文件  f_close(&file);  if(f_open(&file,"upgrade.txt", FA_OPEN_EXISTING |FA_READ) == FR_OK)//assert { if(FR_OK == f_read(&file, read_txt_file_buff, 8, (void *)&numOfOperationBytes)) { if(read_txt_file_buff[0] == 'z' &&read_txt_file_buff[1] == 'q' &&read_txt_file_buff[2] == '7' &&read_txt_file_buff[3] == '7' &&read_txt_file_buff[4] == '7' &&read_txt_file_buff[5] == '7') { //gb_udiskupgradeflag = 1; } } f_close(&file);  } f_mount(0, NULL);  } else//U盤無效 { return(-1); }  f_mount( 0, &fatfs ); res = f_open(&file, "main.bin", FA_OPEN_EXISTING|FA_READ); res = f_read(&file, read_txt_file_buff, 2048, (void *)&numOfOperationBytes); f_close(&file); f_mount(0, NULL);  gb_udiskIsOnLine = 1;//U盤在線 gb_udiskStateChanged = 1; break;  case USH_USR_FS_READ_FILE_SIZE: USBH_USR_ApplicationState = USH_USR_IDLE; if (f_mount( 0, &fatfs ) != FR_OK )  { udiskErrCode = MOUNT_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_open(&file, pFilePath, FA_OPEN_EXISTING | FA_READ); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = OPEN_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); }  fileSize = file.fsize; f_close(&file); f_mount(0, NULL);  udiskErrCode = UDISK_OPERATION_OK;  gb_udiskOperationFinished = 1; break;  case USH_USR_FS_READFILE_PART: USBH_USR_ApplicationState = USH_USR_IDLE; if ( f_mount( 0, &fatfs ) != FR_OK )  { udiskErrCode = MOUNT_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_open(&file, pFilePath, FA_OPEN_EXISTING | FA_READ); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = OPEN_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_lseek(&file,readOffset); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = SEEK_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_read(&file, pFileData, operationLen, (void *)&numOfOperationBytes); //res = f_read(&file, read_txt_file_buff, 1024, (void *)&numOfOperationBytes); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = READ_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } if(operationLen != numOfOperationBytes) { f_close(&file); f_mount(0, NULL);  udiskErrCode = OPERATION_NOT_COMPLETE; gb_udiskOperationFinished = 1; return(-1); } f_close(&file); f_mount(0, NULL);  udiskErrCode = UDISK_OPERATION_OK;  gb_udiskOperationFinished = 1; break;  case USH_USR_FS_WRITEFILE_PART: USBH_USR_ApplicationState = USH_USR_IDLE; if ( f_mount( 0, &fatfs ) != FR_OK )  { udiskErrCode = MOUNT_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_open(&file, pFilePath, FA_OPEN_EXISTING | FA_WRITE); if(res != FR_OK) { udiskErrCode = OPEN_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_lseek(&file,file.fsize);//寫在尾部 if(res != FR_OK) { udiskErrCode = SEEK_FILE_ERROR; gb_udiskOperationFinished = 1; f_close(&file); f_mount(0, NULL);  return(-1); }  res = f_write(&file, pFileData, operationLen, (void *)&numOfOperationBytes); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = WRITE_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } if(operationLen != numOfOperationBytes) { f_close(&file); f_mount(0, NULL);  udiskErrCode = OPERATION_NOT_COMPLETE; gb_udiskOperationFinished = 1; return(-1); } f_close(&file); f_mount(0, NULL);  udiskErrCode = UDISK_OPERATION_OK;  gb_udiskOperationFinished = 1; break; case USH_USR_FS_CREATE_FILE: USBH_USR_ApplicationState = USH_USR_IDLE; if ( f_mount( 0, &fatfs ) != FR_OK )  { udiskErrCode = MOUNT_ERROR; gb_udiskOperationFinished = 1; return(-1); } res = f_open(&file, pFilePath, FA_CREATE_ALWAYS); if(res != FR_OK) { f_close(&file); f_mount(0, NULL);  udiskErrCode = OPEN_FILE_ERROR; gb_udiskOperationFinished = 1; return(-1); } f_close(&file); f_mount(0, NULL);  udiskErrCode = UDISK_OPERATION_OK;  gb_udiskOperationFinished = 1; break;  default: break; } return(0);} 


03

初始化USB HOST及主程序調(diào)用

調(diào)用如下函數(shù)初始化USB Host:

 /* Init Host Library */ USBH_Init(&USB_OTG_Core, #ifdef USE_USB_OTG_FS  USB_OTG_FS_CORE_ID,#else  USB_OTG_HS_CORE_ID,#endif  &USB_Host, &USBH_MSC_cb,  &USR_cb); 

主程序循環(huán)調(diào)用USBH_Process

 /* Host Task handler */ USBH_Process(&USB_OTG_Core, &USB_Host); 


/ The End /

本文主要介紹了在STM32F105平臺上USB HOST MSC驅(qū)動移植和介紹。后續(xù)還會對程序中用到的文件系統(tǒng)FATFS部分進(jìn)行講解。

本文由【嵌入式案例Show】原創(chuàng)出品,未經(jīng)許可,請勿轉(zhuǎn)載


免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護(hù)是驅(qū)動電源設(shè)計中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計成為提升電機(jī)驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗。要解決這一問題,需從設(shè)計、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機(jī)驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉