<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > 關(guān)于ucos中os_tmr.c中的代碼分析

          關(guān)于ucos中os_tmr.c中的代碼分析

          作者: 時間:2016-11-24 來源:網(wǎng)絡(luò) 收藏
          我本身也是個初學(xué)者,喜歡嵌入式而自學(xué)ucos系統(tǒng),ucos是個開源的代碼,短小而又簡單,這是我學(xué)習(xí)的筆記,希望能對喜歡ucos的人有一點幫助,因本人也是初學(xué)者,如有錯誤迎指點。一般的書多是2.5版本,沒有os_tmr.c,所以我寫了關(guān)于這部分代碼的分析。
          我讀ucos.中的os_tmr.c:
          我想這個文件里就是為了寫一個建立在操作系統(tǒng)的定時器,原來我們學(xué)的時鐘節(jié)拍就像cpu總線時鐘脈沖一樣。我們建立的是定時器結(jié)構(gòu)體,在os_tmr.c中有一個函數(shù)OSTmr_Task()這個函數(shù)對定時器結(jié)構(gòu)體的信息進(jìn)行處理,在定時時間到了時候,該定時器中的一個指向回調(diào)函數(shù)的指針就調(diào)用這個回調(diào)節(jié)器函數(shù)進(jìn)行工作,當(dāng)然,你要寫回調(diào)函數(shù),不然就什么也不做,以前做過ucos移植的人知道要寫一個硬件定時器中斷函數(shù),這回要加個OSTmrSignal()這里有個發(fā)送信號。我們學(xué)硬件時知道,用到硬件定時器時要給它一個計算脈沖。這個軟件定時器也要一個計算脈沖。我們建立的定時器結(jié)構(gòu)體都會掛到OSTmrWheelTbl[OS_TMR_CFG_WHEEL_SIZE]上面,OS_TMR_CFG_WHEEL+SIZE是要自己定義的,至于掛到OSTmrWheelTbl[0] 到OSTmrWheelTbl[OS_TMR_CFG_WHEEL_SIZE-1]哪一個上,只要看一下OSTmr_Link (OS_TMR *ptmr, INT8U type)就明白了,而且一個OSTmrWheelTbl[n]上可以掛很多個定時器結(jié)構(gòu)體。
          一、定時器的建立:
          定時器是在我們的應(yīng)用程序中建立的,
          OS_TMR *OSTmrCreate (INT32U dly,
          INT32U period,
          INT8U opt,
          OS_TMR_CALLBACK callback,
          void *callback_arg,
          INT8U *pname,
          INT8U *perr)
          想使用定時器那些函數(shù)要在os_cfg.h中定義 OS_TMR_EN。
          返回值是os_tmr結(jié)構(gòu)體(定義在ucos_ii.h中)。
          typedef struct os_tmr {
          INT8U OSTmrType;
          OS_TMR_CALLBACK OSTmrCallback;
          void *OSTmrCallbackArg;
          void *OSTmrNext;
          void *OSTmrPrev;
          INT32U OSTmrMatch; 當(dāng) OSTmrTime == OSTmrMatch 定時器到時間了。
          INT32U OSTmrDly;
          INT32U OSTmrPeriod;
          #if OS_TMR_CFG_NAME_EN > 0u
          INT8U *OSTmrName;
          #endif
          INT8U OSTmrOpt;
          INT8U OSTmrState;


          } OS_TMR;
          OSTmrType類型定義在ucos_ii.h中:OS_TMR_TYPE
          OS_ARG_CHK_EN要在os_cfg.h中定義一下,來確定要不要一些功能。
          OS_TMR_CFG_NAME_EN要你自己在os_cfg.h中定義,來控制要不要用OSTmrName
          OSTmrState在ucos_ii.h中有以下幾種類型:
          OS_TMR_STATE_UNUSED 不存在這個定時器
          OS_TMR_STATE_RUNNING 這個定時器正在運(yùn)行
          OS_TMR_STATE_COMPLETED這個定時器已經(jīng)跑完了
          OS_TMR_STATE_STOPPED 這個定時器停止了
          參數(shù):
          1. Dly 定時時間,如果是這個定時器只用一次,那么就用這個,如果定時器要反復(fù)用那么它是第一次時用,以后用period。
          2. Period 定時器從復(fù)用時會用到這個作定時時間。
          3. Opt 這里有兩種選項,告訴我們是只用一次還是反復(fù)使用。只用一次OS_TMR_OPT_ONE_SHOT,反復(fù)使用OS_TMR_OPT_PERIODIC。這些定義在ucos_ii.h中。
          定時器選項有五種
          #define OS_TMR_OPT_NONE 0u 沒有選擇
          #define OS_TMR_OPT_ONE_SHOT 1u 定時器不會自動重復(fù)使用
          #define OS_TMR_OPT_PERIODIC 2u 定時器會自動重裝
          #define OS_TMR_OPT_CALLBACK 3u OSTmrStop()中使用,調(diào)用回調(diào)函數(shù),但不帶參數(shù)
          #define OS_TMR_OPT_CALLBACK_ARG 4u 也是OSTmrStop()中使用,調(diào)用回調(diào)函數(shù),但有參數(shù)。
          1. Callback 指向回調(diào)函數(shù)的指針,這個函數(shù)這樣聲明,void mycallback(OS_TMR *ptmr, void p_arg );
          2. Callback_arg 參數(shù)給callback的。
          3. Pname 定時器的名字
          4. Perr 錯誤指針* OS_ERR_NONE 沒有錯誤
          OS_ERR_TMR_INVALID_DLY 無效的定時時間
          OS_ERR_TMR_INVALID_PERIOD 無效的周期
          OS_ERR_TMR_INVALID_OPT 無效的選項
          OS_ERR_TMR_ISR 在中斷中調(diào)用
          OS_ERR_TMR_NON_AVAIL 空的定時器用光了,這個和task一樣意思。
          用到的函數(shù)OSTmr_Alloc()得到一個定時器結(jié)構(gòu)體。
          二、刪除一個定時器,也是在我們的功能函數(shù)中使用,返回為是否成功刪除。
          BOOLEAN OSTmrDel (OS_TMR *ptmr,
          INT8U *perr)
          1. Ptmr 指向定時器結(jié)構(gòu)體。
          2. Perr 指向錯誤的指針。
          這里用到這兩個函數(shù)OSTmr_Unlink(ptmr); 如果是定時器在工作時,要用它解除 OSTmrState= OS_TMR_STATE_STOPPED。
          OSTmr_Free(ptmr); 釋放這個定時器結(jié)構(gòu)體。
          三、得到定時器名字的函數(shù),返回名字的長度。
          INT8U OSTmrNameGet (OS_TMR *ptmr,
          INT8U **pdest, 指向了一個指向定時器名字地址指針的指針。
          INT8U *perr)
          四、定時器還有多長時間溢出。返回還有多長時間溢出。
          INT32U OSTmrRemainGet (OS_TMR *ptmr,
          INT8U *perr)
          五、獲得定時器狀態(tài)的函數(shù),返回狀態(tài)。
          INT8U OSTmrStateGet (OS_TMR *ptmr,
          INT8U *perr)
          六、啟動你的定時器,返回是否成功啟動。 你的應(yīng)用程序使用它
          BOOLEAN OSTmrStart (OS_TMR *ptmr, 要用到OSTmr_Unlink()和OSTmr_Link()先
          INT8U *perr)散 解除,再重新用這個定時器
          七、停止定時器,返回是否成功停止。 你的應(yīng)用程序使用它
          BOOLEAN OSTmrStop (OS_TMR *ptmr,
          INT8U opt,
          void *callback_arg, 這個也是個函數(shù)
          INT8U *perr)
          回調(diào)函數(shù)在這里使用,callback()。
          Opt 為OS_TMR_OPT_NONE不使用回調(diào)函數(shù)。
          OS_TMR_OPT_CALLBACK使用回調(diào)函數(shù)不用參數(shù)。
          OS_TMR_OPT_CALLBACK_ARG要使用參數(shù)。
          八、發(fā)送信號,這個是在timer tick中使用要您寫到ISR中
          INT8U OSTmrSignal (void)返回信號量。
          九、從定時器池中得到一個結(jié)構(gòu)體。在建構(gòu)函數(shù)中用
          static OS_TMR *OSTmr_Alloc (void);
          十、釋放定時器,中刪除函數(shù)中用
          static void OSTmr_Free (OS_TMR *ptmr)
          十一、OSTmr_Init(void),在OSInit()中用。
          十二、static void OSTmr_InitTask (void)在OSTmr_Init中使用。用來建立一個任務(wù)OSTmr_Task()
          十三、OSTmr_Task()這個是調(diào)度你建立的定時器用的,一但定時時間到就調(diào)用回調(diào)函數(shù)。
          我們建立的定時器都進(jìn)入定時器輪盤里OSTmrWheelTbl[],
          十四、static void OSTmr_Link (OS_TMR *ptmr, //OSTmrState = OS_TMR_STATE_RUNNING
          INT8U type)
          Ptmr->OSTmrMatch的確定方法
          if (type == OS_TMR_LINK_PERIODIC) {
          ptmr->OSTmrMatch = ptmr->OSTmrPeriod + OSTmrTime;
          } else {
          if (ptmr->OSTmrDly == 0) {
          ptmr->OSTmrMatch = ptmr->OSTmrPeriod + OSTmrTime;
          } else {
          ptmr->OSTmrMatch = ptmr->OSTmrDly + OSTmrTime;
          }
          }
          掛載定時器時spoke = (INT16U)(ptmr->OSTmrMatch % OS_TMR_CFG_WHEEL_SIZE);
          pspoke = &OSTmrWheelTbl[spoke];這樣確定的置位,當(dāng)OSTmrTime加到和OSTmrMatch相等時一定會來以這個spoke為下標(biāo)的數(shù)組里找該定時器。至于定義一個OSTmrWheelTbl[]而不是把你所有建立的定時器串成一串是怕一起處理浪費(fèi)時間吧,這樣可以一次少處理幾個定時器。
          我想看了OSTmr_Task (void *p_arg)這個函數(shù)的人可能會好奇為什么用那種方法掛載定時器,當(dāng)定時時間到了時會找到OSTmrWheelTbl[]正確的下標(biāo),并在那個OSTmrWheelTbl[ok]里找到該定時器吧。其實你可以算一下,定義OS_TMR_CFG_WHEEL_SIZE=8,然后在OSTmrTime=6時建立一個定時器(假如定時器只工作一次),OSTmrDly=12,那個這個定時器會掛到OSTmrWheelTbl[2]中,當(dāng)OSTmrTime加到18時它就會去
          OSTmrWheelTbl[2]找該定時器。
          十五、static void OSTmr_Unlink (OS_TMR *ptmr)釋放定時器結(jié)構(gòu)體用它把該定時器從定時器輪中卸下。


          關(guān)鍵詞: ucosos_tmrc代碼分

          評論


          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();