上电的过程是:/******************************************************************************* * Function Name : PowerOn * Description : 上电 * Input : None. * Output : None. * Return : USB_SUCCESS. *******************************************************************************/ RESULT PowerOn(void) { u16 wRegVal; USB_Cable_Config(ENABLE); //接上上拉电阻 wRegVal = CNTR_FRES; //设置强制复位 _SetCNTR(wRegVal); wInterrupt_Mask = 0; //先禁止所有的中断 _SetCNTR(wInterrupt_Mask); _SetISTR(0); //清除所有的中断标志 wInterrupt_Mask = CNTR_RESETM | CNTR_SUSPM | CNTR_WKUPM; _SetCNTR(wInterrupt_Mask);//再打开复位中断、挂起中断、唤醒中断屏蔽位 return USB_SUCCESS; }
掉电的过程很简单,先强制复位USB,并清除所有的中断标志,断开上拉电阻,并设置控制进入断电模式。/******************************************************************************* * Function Name : PowerOff * Description : 掉电 * Input : None. * Output : None. * Return : USB_SUCCESS. *******************************************************************************/ RESULT PowerOff() { /* disable all ints and force USB reset */ _SetCNTR(CNTR_FRES); //设置强制复位 _SetISTR(0); //清除所有的中断标志 USB_Cable_Config(DISABLE); //断开上拉电阻 _SetCNTR(CNTR_FRES + CNTR_PDWN); //设置强制复位,并且进入断电模式 return USB_SUCCESS; }
进入挂起状态的过程跟我们平时进入中断服务函数的过程很像,当然要保护现场了。所以usb进入挂起状态,不改变控制寄存器原来的值,只是在原来的上面添加了挂起状态和低功耗状态标志而已。/******************************************************************************* * Function Name : Suspend * Description : 挂起 * Input : None. * Output : None. * Return : USB_SUCCESS. *******************************************************************************/ void Suspend(void) { u16 wCNTR; wCNTR = _GetCNTR(); //读取控制寄存器的值 wCNTR |= CNTR_FSUSP; //添加强制挂起标志 _SetCNTR(wCNTR); //进入挂起模式 wCNTR = _GetCNTR(); //读取控制寄存器的值 wCNTR |= CNTR_LPMODE; //添加低功耗标志 _SetCNTR(wCNTR); //进入低功耗模式 Enter_LowPowerMode(); //进入低功耗模式 }
这个就是唤醒的函数了,比较简单,但是比较难理解的是恢复的各个状态。/******************************************************************************* * Function Name : Resume_Init * Description : 处理唤醒恢复的函数 * Input : None. * Output : None. * Return : USB_SUCCESS. *******************************************************************************/ void Resume_Init(void) { u16 wCNTR; wCNTR = _GetCNTR(); wCNTR &= (~CNTR_LPMODE); _SetCNTR(wCNTR); //进入非低功耗模式 Leave_LowPowerMode(); //离开低功耗模式 _SetCNTR(IMR_MSK); //打开使能全部中断 }
恢复有这么多的状态变化,当然有理由理解下各个状态了。typedef enum _RESUME_STATE { RESUME_EXTERNAL, RESUME_INTERNAL, RESUME_LATER, RESUME_WAIT, RESUME_START, RESUME_ON, RESUME_OFF, RESUME_ESOF } RESUME_STATE;
明显可以看到个状态的转化过程:/******************************************************************************* * Function Name : Resume * Description :这是状态机处理恢复操作和时序。控制是基于Resume结构变量和 * ESOF中断调用该子程序没有改变机状态。控制恢复的状态 * Input : a state machine value (RESUME_STATE) * RESUME_ESOF doesn't change ResumeS.eState allowing * decrementing of the ESOF counter in different states. * Output : None. * Return : None. *******************************************************************************/ void Resume(RESUME_STATE eResumeSetVal) { u16 wCNTR; if (eResumeSetVal != RESUME_ESOF) //如果不是ESOF中断导致的 ResumeS.eState = eResumeSetVal; //ResumeS.eState设置为自己设定的值 switch (ResumeS.eState) { case RESUME_EXTERNAL: //RESUME_EXTERNAL 外部恢复 Resume_Init(); ResumeS.eState = RESUME_OFF; break; case RESUME_INTERNAL: //RESUME_INTERNAL 内部恢复 Resume_Init(); ResumeS.eState = RESUME_START; break; case RESUME_LATER: //RESUME_LATER 定时恢复 ResumeS.bESOFcnt = 2; ResumeS.eState = RESUME_WAIT; break; case RESUME_WAIT: //RESUME_WAIT 等待计时结束 ResumeS.bESOFcnt--; if (ResumeS.bESOFcnt == 0) ResumeS.eState = RESUME_START; break; case RESUME_START: //RESUME_START 开始恢复 wCNTR = _GetCNTR(); wCNTR |= CNTR_RESUME; _SetCNTR(wCNTR); //设置唤醒请求位,将向PC主机发送唤醒请求 ResumeS.eState = RESUME_ON; ResumeS.bESOFcnt = 10; //定时10ms,如果在这点时间内保持有效,主机将对USB模块进行唤醒操作 break; case RESUME_ON: //RESUME_ON ResumeS.bESOFcnt--; //计时中 if (ResumeS.bESOFcnt == 0) //计时时间到了 { wCNTR = _GetCNTR(); wCNTR &= (~CNTR_RESUME); //清除唤醒请求标志位 _SetCNTR(wCNTR); ResumeS.eState = RESUME_OFF; } break; case RESUME_OFF: //RESUME_OFF case RESUME_ESOF: //RESUME_ESOF default: ResumeS.eState = RESUME_OFF; break; } }