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

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

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ǔ)。在實(shí)際使用中,USB OTG是USB Device和USB Host的底層驅(qū)動(dòng)。本例需要移植OTG+HOST。(現(xiàn)在可以不需要移植了,通過STM32CubeMX直接由圖形界面簡(jiǎn)單配置下,生成初始化代碼,并對(duì)外設(shè)做了進(jìn)一步的抽象,讓開發(fā)人員更只專注應(yīng)用的開發(fā)。搞嵌入式還是要了解底層驅(qū)動(dòng),對(duì)吧?)

01

驅(qū)動(dòng)文件移植

移植文件如下圖:

1.1 OTG和HOST驅(qū)動(dòng)文件簡(jiǎn)要說明

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

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

1.3 用戶文件介紹

通常所有的用戶代碼均在以上文件做修改即可,只有少數(shù)特殊處理會(huì)設(shè)計(jì)到驅(qū)動(dòng)程序源碼的修改。在配置好后,一般來說,所有修改均在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時(shí),非周期性發(fā)送的FIFO的大小 #define TXH_P_HS_FIFOSIZ 256 //設(shè)置高速模式下,作為USB Host時(shí),周期性發(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時(shí),非周期性發(fā)送的FIFO的大小 #define TXH_P_FS_FIFOSIZ 96 //設(shè)置全速模式下,作為USB Host時(shí),周期性發(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 //支持端點(diǎn)的最大數(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ù)實(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)檢測(cè)到設(shè)備斷開連接事件時(shí)以及未發(fā)生錯(cuò)誤時(shí)也會(huì)進(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)一個(gè)設(shè)備連接時(shí),核心進(jìn)入這個(gè)狀態(tài)。 當(dāng)一個(gè)設(shè)備被檢測(cè)到時(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é)束時(shí),選擇默認(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: //這是一個(gè)中間狀態(tài),它遵循枚舉并包括等待用戶輸入以啟動(dòng)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ū)動(dòng)程序接管,并調(diào)用類請(qǐng)求狀態(tài)機(jī)以處理所有初始類控制請(qǐng)求。 完成所需的類請(qǐng)求后,內(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)移時(shí)就會(huì)進(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ù)的錯(cuò)誤,就會(huì)進(jìn)入此狀態(tài)。 在這種情況下,調(diào)用用戶回調(diào)函數(shù)(例如,顯示未恢復(fù)的錯(cuò)誤消息)。 然后主機(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盤讀寫操作的上層實(shí)現(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); //寫一個(gè)測(cè)試文件  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平臺(tái)上USB HOST MSC驅(qū)動(dòng)移植和介紹。后續(xù)還會(huì)對(duì)程序中用到的文件系統(tǒng)FATFS部分進(jìn)行講解。

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


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

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

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉