在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

鴻蒙內核源碼分析:進程間異步解耦大數據傳遞

鴻蒙系統HarmonyOS ? 來源:my.oschina ? 作者:鴻蒙內核源碼分析 ? 2021-04-24 11:32 ? 次閱讀

基本概念

隊列又稱消息隊列,是一種常用于任務間通信的數據結構。隊列接收來自任務或中斷的不固定長度消息,并根據不同的接口確定傳遞的消息是否存放在隊列空間中。

任務能夠從隊列里面讀取消息,當隊列中的消息為空時,掛起讀取任務;當隊列中有新消息時,掛起的讀取任務被喚醒并處理新消息。任務也能夠往隊列里寫入消息,當隊列已經寫滿消息時,掛起寫入任務;當隊列中有空閑消息節點時,掛起的寫入任務被喚醒并寫入消息。如果將讀隊列和寫隊列的超時時間設置為0,則不會掛起任務,接口會直接返回,這就是非阻塞模式。

消息隊列提供了異步處理機制,允許將一個消息放入隊列,但不立即處理。同時隊列還有緩沖消息的作用。

隊列用于任務間通信,可以實現消息的異步處理。同時消息的發送方和接收方不需要彼此聯系,兩者間是解耦的。

隊列特性

消息以先進先出的方式排隊,支持異步讀寫。

讀隊列和寫隊列都支持超時機制。

每讀取一條消息,就會將該消息節點設置為空閑。

發送消息類型由通信雙方約定,可以允許不同長度(不超過隊列的消息節點大小)的消息。

一個任務能夠從任意一個消息隊列接收和發送消息。

多個任務能夠從同一個消息隊列接收和發送消息。

創建隊列時所需的隊列空間,默認支持接口內系統自行動態申請內存的方式,同時也支持將用戶分配的隊列空間作為接口入參傳入的方式。

消息隊列長什么樣?

#ifndef LOSCFG_BASE_IPC_QUEUE_LIMIT
#define LOSCFG_BASE_IPC_QUEUE_LIMIT 1024 //隊列個數
#endif
LITE_OS_SEC_BSS LosQueueCB *g_allQueue = NULL;//消息隊列池
LITE_OS_SEC_BSS STATIC LOS_DL_LIST g_freeQueueList;//空閑隊列鏈表,管分配的,需要隊列從這里申請

typedef struct {
    UINT8 *queueHandle; /**< Pointer to a queue handle */	//指向隊列句柄的指針
    UINT16 queueState; /**< Queue state */	//隊列狀態
    UINT16 queueLen; /**< Queue length */	//隊列中消息總數的上限值,由創建時確定,不再改變
    UINT16 queueSize; /**< Node size */		//消息節點大小,由創建時確定,不再改變,即定義了每個消息長度的上限.
    UINT32 queueID; /**< queueID */			//隊列ID
    UINT16 queueHead; /**< Node head */		//消息頭節點位置(數組下標)
    UINT16 queueTail; /**< Node tail */		//消息尾節點位置(數組下標)
    UINT16 readWriteableCnt[OS_QUEUE_N_RW]; /**< Count of readable or writable resources, 0:readable, 1:writable */
											//隊列中可寫或可讀消息數,0表示可讀,1表示可寫
    LOS_DL_LIST readWriteList[OS_QUEUE_N_RW]; /**< the linked list to be read or written, 0:readlist, 1:writelist */
											//掛的都是等待讀/寫消息的任務鏈表,0表示讀消息的鏈表,1表示寫消息的任務鏈表
    LOS_DL_LIST memList; /**< Pointer to the memory linked list */	//@note_why 這里尚未搞明白是啥意思 ,是共享內存嗎?
} LosQueueCB;//讀寫隊列分離

解讀

和進程,線程,定時器一樣,消息隊列也由全局統一的消息隊列池管理,池有多大?默認是1024

鴻蒙內核對消息的總個數有限制,queueLen消息總數的上限,在創建隊列的時候需指定,不能更改.

對每個消息的長度也有限制,queueSize規定了消息的大小,也是在創建的時候指定.

為啥要指定總個數和每個的總大小,是因為內核一次性會把隊列的總內存(queueLen*queueSize)申請下來,確保不會出現后續使用過程中內存不夠的問題出現,但同時也帶來了內存的浪費,因為很可能大部分時間隊列并沒有跑滿.

隊列的讀取由queueHead,queueTail管理,Head表示隊列中被占用的消息節點的起始位置。Tail表示被占用的消息節點的結束位置,也是空閑消息節點的起始位置。隊列剛創建時,Head和Tail均指向隊列起始位置

寫隊列時,根據readWriteableCnt[1]判斷隊列是否可以寫入,不能對已滿(readWriteableCnt[1]為0)隊列進行寫操作。寫隊列支持兩種寫入方式:向隊列尾節點寫入,也可以向隊列頭節點寫入。尾節點寫入時,根據Tail找到起始空閑消息節點作為數據寫入對象,如果Tail已經指向隊列尾部則采用回卷方式。頭節點寫入時,將Head的前一個節點作為數據寫入對象,如果Head指向隊列起始位置則采用回卷方式。

讀隊列時,根據readWriteableCnt[0]判斷隊列是否有消息需要讀取,對全部空閑(readWriteableCnt[0]為0)隊列進行讀操作會引起任務掛起。如果隊列可以讀取消息,則根據Head找到最先寫入隊列的消息節點進行讀取。如果Head已經指向隊列尾部則采用回卷方式。

刪除隊列時,根據隊列ID找到對應隊列,把隊列狀態置為未使用,把隊列控制塊置為初始狀態。如果是通過系統動態申請內存方式創建的隊列,還會釋放隊列所占內存。

留意readWriteList,這又是兩個雙向鏈表, 雙向鏈表是內核最重要的結構體,牢牢的寄生在宿主結構體上.readWriteList上掛的是未來讀/寫消息隊列的任務列表.

初始化隊列

LITE_OS_SEC_TEXT_INIT UINT32 OsQueueInit(VOID)//消息隊列模塊初始化
{
    LosQueueCB *queueNode = NULL;
    UINT32 index;
    UINT32 size;

    size = LOSCFG_BASE_IPC_QUEUE_LIMIT * sizeof(LosQueueCB);//支持1024個IPC隊列
    /* system resident memory, don't free */
    g_allQueue = (LosQueueCB *)LOS_MemAlloc(m_aucSysMem0, size);//常駐內存
    if (g_allQueue == NULL) {
        return LOS_ERRNO_QUEUE_NO_MEMORY;
    }
    (VOID)memset_s(g_allQueue, size, 0, size);//清0
    LOS_ListInit(&g_freeQueueList);//初始化空閑鏈表
    for (index = 0; index < LOSCFG_BASE_IPC_QUEUE_LIMIT; index++) {//循環初始化每個消息隊列
        queueNode = ((LosQueueCB *)g_allQueue) + index;//一個一個來
        queueNode->queueID = index;//這可是 隊列的身份證
        LOS_ListTailInsert(&g_freeQueueList, &queueNode->readWriteList[OS_QUEUE_WRITE]);//通過寫節點掛到空閑隊列鏈表上
    }//這里要注意是用 readWriteList 掛到 g_freeQueueList鏈上的,所以要通過 GET_QUEUE_LIST 來找到 LosQueueCB

    if (OsQueueDbgInitHook() != LOS_OK) {//調試隊列使用的.
        return LOS_ERRNO_QUEUE_NO_MEMORY;
    }
    return LOS_OK;
}

解讀

初始隊列模塊,對幾個全局變量賦值,創建消息隊列池,所有池都是常駐內存,關于池后續有專門的文章整理,到目前為止已經解除到了進程池,任務池,定時器池,隊列池,==

將LOSCFG_BASE_IPC_QUEUE_LIMIT個隊列掛到空閑鏈表g_freeQueueList上,供后續分配和回收.熟悉內核全局資源管理的對這種方式應該不會再陌生.

創建隊列

//創建一個隊列,根據用戶傳入隊列長度和消息節點大小來開辟相應的內存空間以供該隊列使用,參數queueID帶走隊列ID
LITE_OS_SEC_TEXT_INIT UINT32 LOS_QueueCreate(CHAR *queueName, UINT16 len, UINT32 *queueID,
                                             UINT32 flags, UINT16 maxMsgSize)
{
    LosQueueCB *queueCB = NULL;
    UINT32 intSave;
    LOS_DL_LIST *unusedQueue = NULL;
    UINT8 *queue = NULL;
    UINT16 msgSize;

    (VOID)queueName;
    (VOID)flags;

    if (queueID == NULL) {
        return LOS_ERRNO_QUEUE_CREAT_PTR_NULL;
    }

    if (maxMsgSize > (OS_NULL_SHORT - sizeof(UINT32))) {// maxMsgSize上限 為啥要減去 sizeof(UINT32) ,因為前面存的是隊列的大小
        return LOS_ERRNO_QUEUE_SIZE_TOO_BIG;
    }

    if ((len == 0) || (maxMsgSize == 0)) {
        return LOS_ERRNO_QUEUE_PARA_ISZERO;
    }

    msgSize = maxMsgSize + sizeof(UINT32);//總size = 消息體內容長度 + 消息大小(UINT32) 
    /*
     * Memory allocation is time-consuming, to shorten the time of disable interrupt,
     * move the memory allocation to here.
     *///內存分配非常耗時,為了縮短禁用中斷的時間,將內存分配移到此處,用的時候分配隊列內存
    queue = (UINT8 *)LOS_MemAlloc(m_aucSysMem1, (UINT32)len * msgSize);//從系統內存池中分配,由這里提供讀寫隊列的內存
    if (queue == NULL) {//這里是一次把隊列要用到的所有最大內存都申請下來了,能保證不會出現后續使用過程中內存不夠的問題出現
        return LOS_ERRNO_QUEUE_CREATE_NO_MEMORY;//調用處有 OsSwtmrInit sys_mbox_new DoMqueueCreate ==
    }

    SCHEDULER_LOCK(intSave);
    if (LOS_ListEmpty(&g_freeQueueList)) {//沒有空余的隊列ID的處理,注意軟時鐘定時器是由 g_swtmrCBArray統一管理的,里面有正在使用和可分配空閑的隊列
        SCHEDULER_UNLOCK(intSave);//g_freeQueueList是管理可用于分配的隊列鏈表,申請消息隊列的ID需要向它要
        OsQueueCheckHook();
        (VOID)LOS_MemFree(m_aucSysMem1, queue);//沒有就要釋放 queue申請的內存
        return LOS_ERRNO_QUEUE_CB_UNAVAILABLE;
    }

    unusedQueue = LOS_DL_LIST_FIRST(&g_freeQueueList);//找到一個沒有被使用的隊列
    LOS_ListDelete(unusedQueue);//將自己從g_freeQueueList中摘除, unusedQueue只是個 LOS_DL_LIST 結點.
    queueCB = GET_QUEUE_LIST(unusedQueue);//通過unusedQueue找到整個消息隊列(LosQueueCB)
    queueCB->queueLen = len;	//隊列中消息的總個數,注意這個一旦創建是不能變的.
    queueCB->queueSize = msgSize;//消息節點的大小,注意這個一旦創建也是不能變的.
    queueCB->queueHandle = queue;	//隊列句柄,隊列內容存儲區. 
    queueCB->queueState = OS_QUEUE_INUSED;	//隊列狀態使用中
    queueCB->readWriteableCnt[OS_QUEUE_READ] = 0;//可讀資源計數,OS_QUEUE_READ(0):可讀.
    queueCB->readWriteableCnt[OS_QUEUE_WRITE] = len;//可些資源計數 OS_QUEUE_WRITE(1):可寫, 默認len可寫.
    queueCB->queueHead = 0;//隊列頭節點
    queueCB->queueTail = 0;//隊列尾節點
    LOS_ListInit(&queueCB->readWriteList[OS_QUEUE_READ]);//初始化可讀隊列鏈表
    LOS_ListInit(&queueCB->readWriteList[OS_QUEUE_WRITE]);//初始化可寫隊列鏈表
    LOS_ListInit(&queueCB->memList);//

    OsQueueDbgUpdateHook(queueCB->queueID, OsCurrTaskGet()->taskEntry);//在創建或刪除隊列調試信息時更新任務條目
    SCHEDULER_UNLOCK(intSave);

    *queueID = queueCB->queueID;//帶走隊列ID
    return LOS_OK;
}

解讀

創建和初始化一個LosQueueCB

動態分配內存來保存消息內容,LOS_MemAlloc(m_aucSysMem1, (UINT32)len * msgSize);

msgSize = maxMsgSize + sizeof(UINT32);頭四個字節放消息的長度,但消息最大長度不能超過maxMsgSize

readWriteableCnt記錄讀/寫隊列的數量,獨立計算

readWriteList掛的是等待讀取隊列的任務鏈表 將在OsTaskWait(&queueCB->readWriteList[readWrite], timeout, TRUE);中將任務掛到鏈表上.

關鍵函數OsQueueOperate

隊列的讀寫都要經過OsQueueOperate

o4YBAGCDkTmAUbFPAADjqnMUAgo404.png

/************************************************
隊列操作.是讀是寫由operateType定
本函數是消息隊列最重要的一個函數,可以分析出讀取消息過程中
發生的細節,涉及任務的喚醒和阻塞,阻塞鏈表任務的相互提醒.
************************************************/
UINT32 OsQueueOperate(UINT32 queueID, UINT32 operateType, VOID *bufferAddr, UINT32 *bufferSize, UINT32 timeout)
{
    LosQueueCB *queueCB = NULL;
    LosTaskCB *resumedTask = NULL;
    UINT32 ret;
    UINT32 readWrite = OS_QUEUE_READ_WRITE_GET(operateType);//獲取讀/寫操作標識
    UINT32 intSave;

    SCHEDULER_LOCK(intSave);
    queueCB = (LosQueueCB *)GET_QUEUE_HANDLE(queueID);//獲取對應的隊列控制塊
    ret = OsQueueOperateParamCheck(queueCB, queueID, operateType, bufferSize);//參數檢查
    if (ret != LOS_OK) {
        goto QUEUE_END;
    }

    if (queueCB->readWriteableCnt[readWrite] == 0) {//根據readWriteableCnt判斷隊列是否有消息讀/寫
        if (timeout == LOS_NO_WAIT) {//不等待直接退出
            ret = OS_QUEUE_IS_READ(operateType) ? LOS_ERRNO_QUEUE_ISEMPTY : LOS_ERRNO_QUEUE_ISFULL;
            goto QUEUE_END;
        }

        if (!OsPreemptableInSched()) {//不支持搶占式調度
            ret = LOS_ERRNO_QUEUE_PEND_IN_LOCK;
            goto QUEUE_END;
        }
		//任務等待,這里很重要啊,將自己從就緒列表摘除,讓出了CPU并發起了調度,并掛在readWriteList[readWrite]上,掛的都等待讀/寫消息的task
        ret = OsTaskWait(&queueCB->readWriteList[readWrite], timeout, TRUE);//任務被喚醒后會回到這里執行,什么時候會被喚醒?當然是有消息的時候!
        if (ret == LOS_ERRNO_TSK_TIMEOUT) {//喚醒后如果超時了,返回讀/寫消息失敗
            ret = LOS_ERRNO_QUEUE_TIMEOUT;
            goto QUEUE_END;//
        }
    } else {
        queueCB->readWriteableCnt[readWrite]--;//對應隊列中計數器--,說明一條消息只能被讀/寫一次
    }

    OsQueueBufferOperate(queueCB, operateType, bufferAddr, bufferSize);//發起讀或寫隊列操作

    if (!LOS_ListEmpty(&queueCB->readWriteList[!readWrite])) {//如果還有任務在排著隊等待讀/寫入消息(當時不能讀/寫的原因有可能當時隊列滿了==)
        resumedTask = OS_TCB_FROM_PENDLIST(LOS_DL_LIST_FIRST(&queueCB->readWriteList[!readWrite]));//取出要讀/寫消息的任務
        OsTaskWake(resumedTask);//喚醒任務去讀/寫消息啊
        SCHEDULER_UNLOCK(intSave);
        LOS_MpSchedule(OS_MP_CPU_ALL);//讓所有CPU發出調度申請,因為很可能那個要讀/寫消息的隊列是由其他CPU執行
        LOS_Schedule();//申請調度
        return LOS_OK;
    } else {
        queueCB->readWriteableCnt[!readWrite]++;//對應隊列讀/寫中計數器++
    }

QUEUE_END:
    SCHEDULER_UNLOCK(intSave);
    return ret;
}

解讀

queueID指定操作消息隊列池中哪個消息隊列

operateType表示本次是是讀/寫消息

bufferAddr,bufferSize表示如果讀操作,用buf接走數據,如果寫操作,將buf寫入隊列.

timeout只用于當隊列中沒有讀/寫內容時的等待.

當讀消息時發現隊列中沒有可讀的消息,此時timeout決定是否將任務掛入等待讀隊列任務鏈表

當寫消息時發現隊列中沒有空間用于寫的消息,此時timeout決定是否將任務掛入等待寫隊列任務鏈表

if (!LOS_ListEmpty(&queueCB->readWriteList[!readWrite]))最有意思的是這行代碼.

在一次讀消息完成后會立即喚醒寫隊列任務鏈表的任務,因為讀完了就有了剩余空間,等待寫隊列的任務往往是因為沒有空間而進入等待狀態.

在一次寫消息完成后會立即喚醒讀隊列任務鏈表的任務,因為寫完了隊列就有了新消息,等待讀隊列的任務往往是因為隊列中沒有消息而進入等待狀態.

編程實例

創建一個隊列,兩個任務。任務1調用寫隊列接口發送消息,任務2通過讀隊列接口接收消息。

通過LOS_TaskCreate創建任務1和任務2。

通過LOS_QueueCreate創建一個消息隊列。

在任務1 send_Entry中發送消息。

在任務2 recv_Entry中接收消息。

通過LOS_QueueDelete刪除隊列。

#include "los_task.h"
#include "los_queue.h"

static UINT32 g_queue;
#define BUFFER_LEN 50

VOID send_Entry(VOID)
{
    UINT32 i = 0;
    UINT32 ret = 0;
    CHAR abuf[] = "test is message x";
    UINT32 len = sizeof(abuf);

    while (i < 5) {
        abuf[len -2] = '0' + i;
        i++;

        ret = LOS_QueueWriteCopy(g_queue, abuf, len, 0);
        if(ret != LOS_OK) {
            dprintf("send message failure, error: %x\n", ret);
        }

        LOS_TaskDelay(5);
    }
}

VOID recv_Entry(VOID)
{
    UINT32 ret = 0;
    CHAR readBuf[BUFFER_LEN] = {0};
    UINT32 readLen = BUFFER_LEN;

    while (1) {
        ret = LOS_QueueReadCopy(g_queue, readBuf, &readLen, 0);
        if(ret != LOS_OK) {
            dprintf("recv message failure, error: %x\n", ret);
            break;
        }

        dprintf("recv message: %s\n", readBuf);
        LOS_TaskDelay(5);
    }

    while (LOS_OK != LOS_QueueDelete(g_queue)) {
        LOS_TaskDelay(1);
    }

    dprintf("delete the queue success!\n");
}

UINT32 Example_CreateTask(VOID)
{
    UINT32 ret = 0; 
    UINT32 task1, task2;
    TSK_INIT_PARAM_S initParam;

    initParam.pfnTaskEntry = (TSK_ENTRY_FUNC)send_Entry;
    initParam.usTaskPrio = 9;
    initParam.uwStackSize = LOS_TASK_MIN_STACK_SIZE;
    initParam.pcName = "sendQueue";
#ifdef LOSCFG_KERNEL_SMP
    initParam.usCpuAffiMask = CPUID_TO_AFFI_MASK(ArchCurrCpuid());
#endif
    initParam.uwResved = LOS_TASK_STATUS_DETACHED;

    LOS_TaskLock();
    ret = LOS_TaskCreate(&task1, &initParam);
    if(ret != LOS_OK) {
        dprintf("create task1 failed, error: %x\n", ret);
        return ret;
    }

    initParam.pcName = "recvQueue";
    initParam.pfnTaskEntry = (TSK_ENTRY_FUNC)recv_Entry;
    ret = LOS_TaskCreate(&task2, &initParam);
    if(ret != LOS_OK) {
        dprintf("create task2 failed, error: %x\n", ret);
        return ret;
    }

    ret = LOS_QueueCreate("queue", 5, &g_queue, 0, BUFFER_LEN);
    if(ret != LOS_OK) {
        dprintf("create queue failure, error: %x\n", ret);
    }

    dprintf("create the queue success!\n");
    LOS_TaskUnlock();
    return ret;
}

結果驗證

create the queue success!
recv message: test is message 0
recv message: test is message 1
recv message: test is message 2
recv message: test is message 3
recv message: test is message 4
recv message failure, error: 200061d
delete the queue success!

總結

消息隊列解決任務間大數據的傳遞

以一種異步,解耦的方式實現任務通訊

全局由消息隊列池統一管理

在創建消息隊列時申請內存塊存儲消息內存.

讀/寫操作統一管理,分開執行,A任務讀/寫完消息后會立即喚醒等待寫/讀的B任務.

編輯:hfy

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 異步處理
    +關注

    關注

    0

    文章

    7

    瀏覽量

    6561
收藏 人收藏

    評論

    相關推薦

    鴻蒙內核源碼Task/線程技術分析

    、使用內存空間等系統資源,并獨立于其它線程運行。 鴻蒙內核每個進程內的線程獨立運行、獨立調度,當前進程內線程的調度不受其它進程內線程的影響。
    的頭像 發表于 10-18 10:42 ?2244次閱讀
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>源碼</b>Task/線程技術<b class='flag-5'>分析</b>

    【HarmonyOS】鴻蒙內核源碼分析(調度機制篇)

    意義上所理解的線程呢。狹義上的后續有 鴻蒙內核源碼分析(啟動過程篇) 來說明。不知道大家有沒有這種體會,學一個東西的過程中要接觸很多新概念,尤其像 Java/android 的生態,概
    發表于 10-14 14:00

    鴻蒙源碼分析系列(總目錄) | 給HarmonyOS源碼逐行加上中文注釋

    內核源碼分析(時鐘管理篇) | 觸發調度最大的源動力|-鴻蒙內核源碼
    發表于 11-20 11:24

    鴻蒙內核源碼分析(調度機制篇):Task是如何被調度執行的

    (); 就是設置啟動任務,但此時啥都還沒開始呢,Kprocess 進程都沒創建,怎么會有大家一般意義上所理解的線程呢。狹義上的后續有 鴻蒙內核源碼
    發表于 11-23 10:53

    鴻蒙內核源碼分析(進程管理篇):進程內核的資源管理單元

    ;為什么會這樣?另外兩個爸爸對應的PID是 1和2,那進程池里的0號進程又去哪里了呢?全部文章進入 >> 鴻蒙系統源碼分析(總目錄) 查看
    發表于 11-24 11:23

    HarmonyOS 內核源碼分析(下)—電子書上線啦!

    的第十六章 進程如何異步傳遞大數據第十七章 是
    發表于 04-01 17:33

    HarmonyOS內核源碼分析(下)

    章 任務一對多和多對多的同步方案第十四章 內核最高優先級任務是誰第十五章 內核是如何描述CPU的第十六章 進程如何
    發表于 04-02 15:56

    鴻蒙分布式任務調度——數據傳遞

    鴻蒙分布式任務調度之數據傳遞
    發表于 06-12 17:29

    OpenHarmony進程是如何傳遞大數據

    (OsQueueDbgInitHook() != LOS_OK) {//調試隊列使用的.return LOS_ERRNO_QUEUE_NO_MEMORY;}return LOS_OK;}總結:消息隊列解決任務大數據傳遞,以
    發表于 05-23 17:13

    鴻蒙內核進程為何要通訊?

    鴻蒙內核默認支持 64個進程和128個任務,由進程池和任務池統一管理.內核設計盡量不去打擾它們,讓各自過好各自的日子, 但大家畢竟在一口鍋里
    的頭像 發表于 04-24 11:28 ?2756次閱讀
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>進程</b><b class='flag-5'>間</b>為何要通訊?

    鴻蒙內核源碼分析:task是內核調度的單元

    從系統的角度看,線程是競爭系統資源的最小運行單元。線程可以使用或等待CPU、使用內存空間等系統資源,并獨立于其它線程運行。 鴻蒙內核每個進程內的線程獨立運行、獨立調度,當前進程內線程
    發表于 11-23 15:51 ?22次下載
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>源碼</b><b class='flag-5'>分析</b>:task是<b class='flag-5'>內核</b>調度的單元

    鴻蒙內核源碼分析進程和Task的就緒隊列對調度的作用

    鴻蒙內核代碼中有兩個源文件是關于隊列的,一個是用于調度的隊列,另一個是用于線程通訊的IPC隊列。 鴻蒙內核
    發表于 11-23 15:48 ?31次下載
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>源碼</b><b class='flag-5'>分析</b>:<b class='flag-5'>進程</b>和Task的就緒隊列對調度的作用

    鴻蒙內核源碼分析進程內核的資源管理單元

    從系統的角度看,進程是資源管理單元。進程可以使用或等待CPU、使用內存空間等系統資源,并獨立于其它進程運行。OpenHarmony內核進程
    發表于 11-24 17:52 ?23次下載
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>源碼</b><b class='flag-5'>分析</b>:<b class='flag-5'>進程</b>是<b class='flag-5'>內核</b>的資源管理單元

    鴻蒙內核源碼分析內核最重要結構體

    為何鴻蒙內核源碼分析系列開篇就說 LOS_DL_LIST ? 因為它在鴻蒙 LOS 內核中無處
    發表于 11-24 17:54 ?35次下載
    <b class='flag-5'>鴻蒙</b><b class='flag-5'>內核</b><b class='flag-5'>源碼</b><b class='flag-5'>分析</b> :<b class='flag-5'>內核</b>最重要結構體

    華為鴻蒙系統內核源碼分析上冊

    鴻蒙內核源碼注釋中文版【 Gitee倉】給 Harmoηy○S源碼逐行加上中文注解,詳細闡述設計細節,助你快速精讀 Harmonyos內核源碼
    發表于 04-09 14:40 ?17次下載
    主站蜘蛛池模板: 九九天天影视| 色噜噜狠狠色综合中文字幕| 成zzzwww日本免费| 日本黄色大片免费| 亚洲福利二区| 欧亚精品卡一卡二卡三| 男人扒开美女尿口无遮挡图片| 欧美人成一本免费观看视频| a黄网站| 热久在线| 伊人久久大香线蕉综合bd高清| 午夜爱爱小视频| 日日噜噜噜夜夜爽爽狠狠图片| 一级aaa毛片| 天堂中文在线网| 欧美日韩一区二区三区视频| 日本成人资源| 久久国产精品免费网站| 久青草免费在线视频| 国产精品国产三级国产在线观看| 中文字幕在线永久在线视频2020| 天堂网资源www| 射久久| 免费福利午夜影视网| 婷婷亚洲五月琪琪综合| 欧美aaaaa| 四虎地址8848最新章节| 四虎最新免费网址| 色月| 日本三级欧美三级香港黄| 就去色综合| 一本到视频在线| 午夜一级毛片看看| 国产黄mmd在线观看免费| 性叉叉| 天天摸天天添人人澡| 久久久精品免费视频| 午夜插| 在线观看一区二区三区视频| 久久久久国产一级毛片高清板| 亚洲成人在线播放|