女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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

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

3天內不再提示

整體thread模型的設計與實現

FPGA之家 ? 來源:CSDN ? 作者:周嘉祺 ? 2021-03-29 14:39 ? 次閱讀

SPDK Thread 模型是SPDK誕生以來十分重要的模塊,它的設計確保了spdk應用的無鎖化編程模型,本文基于spdk最新的release 19.07版本介紹了整體thread模型的設計與實現,并詳細分析了NVMe-oF的使用案例。

P1

SPDK Thread 模型設計與實現

Reactor – 單個CPU Core抽象,主要包含了:

Lcore對應的CPU Core id

Threads在該核心下的線程

Events 這是一個spdk ring,用于事件傳遞接收

Thread – 線程,但它是spdk抽象出來的線程,主要包含了:

io_channels資源的抽象,可以是bdev,也可以是具體的tgt

tailq 線程隊列,用于連接下一個線程

name 線程的名稱

Stats 用于計時統計閑置和忙時時間的

active_pollers 輪詢使用的poller,非定時

timer_pollers 定時的poller

messages 這是一個spdk ring,用于消息傳遞接收

msg_cache 事件的緩存

1.1 Reactor

對象g_reactor_state有五個狀態對應了應用中reactors運行運行狀態,

enum spdk_reactor_state {

SPDK_REACTOR_STATE_INVALID = 0,

SPDK_REACTOR_STATE_INITIALIZED = 1,

SPDK_REACTOR_STATE_RUNNING = 2,

SPDK_REACTOR_STATE_EXITING = 3,

SPDK_REACTOR_STATE_SHUTDOWN = 4,

};

初始情況下是:

SPDK_REACTOR_STATE_INVALID狀態,在spdk app(任意一個target,比如nvmf_tgt)啟動時,即調用了spdk_app_start方法,會調用spdk_reactors_init,在這個方法中將會初始化所有需要被初始化的reactors(可以在配置文件中指定需要使用的Core,CPU Core 和reactor是一對一的)。并且會將g_reactor_state設置為SPDK_REACTOR_STATE_INITIALIZED。具體代碼如下:

Int spdk_reactors_init(void)

{

// 初始化所有的event mempool

g_spdk_event_mempool = spdk_mempool_create(…);

// 為g_reactors分配內存,g_reactors是一個數組,管理了所有的reactors

posix_memalign((void **)&g_reactors, 64, (last_core + 1) * sizeof(struct spdk_reactor));

// 這里設置了reactor創建線程的方法,之后需要初始化線程的時候將會調用該方法

spdk_thread_lib_init(spdk_reactor_schedule_thread, sizeof(struct spdk_lw_thread));

// 對于每一個啟動的reactor,將會初始化它們

// 初始化reactor過程,即為綁定lcore,初始化spdk ring、threads,對rusage無操作

SPDK_ENV_FOREACH_CORE(i) {

reactor = spdk_reactor_get(i);

spdk_reactor_construct(reactor, i);

}

// 設置好狀態返回

g_reactor_state = SPDK_REACTOR_STATE_INITIALIZED;

return 0;

}

在進入SPDK_REACTOR_STATE_INITIALIZED狀態且spdk_app_start在創建了自己的線程并綁定到了reactors后,會調用spdk_reactors_start方法并將g_reactor_state設置為SPDK_REACTOR_STATE_RUNNING狀態并會創建所有reactor的線程且輪詢。

Void spdk_reactors_start(void) {

SPDK_ENV_FOREACH_CORE(i) {

if (i != current_core) { // 在非master reactor中

reactor = spdk_reactor_get(i); // 得到相應的reactor

// 設置好線程創建后的一個消息,該消息為輪詢函數

rc = spdk_env_thread_launch_pinned(reactor-》lcore, _spdk_reactor_run, reactor);

// reactor創建好線程并且會自動執行第一個消息

spdk_thread_create(thread_name, tmp_cpumask);

}

}

// 當前CPU core得到reactor,并且開始輪詢

reactor = spdk_reactor_get(current_core);

_spdk_reactor_run(reactor);

}

之前提到spdk_reactors_init方法中調用了spdk_thread_lib_init方法傳入了創建thread的spdk_reactor_schedule_thread方法,在調用spdk_thread_create會回調該方法。這個方法它主要的功能就是告訴這個新創建的線程綁定創建該線程的reactor。

spdk_reactor_schedule_thread(struct spdk_thread *thread)

{

// 得到該線程設置的cpu mask

cpumask = spdk_thread_get_cpumask(thread);

for (i = 0; i 《 spdk_env_get_core_count(); i++) {

…。 // 遍歷cpu core

// 通過cpu mask找到對應的核心,并產生event

if (spdk_cpuset_get_cpu(cpumask, core)) {

evt = spdk_event_allocate(core, _schedule_thread, lw_thread, NULL);

break;

}

}

// 傳遞該event,即對應的reatcor會調用_schedule_thread方法,

spdk_event_call(evt);

}

_schedule_thread(void *arg1, void *arg2)

{

struct spdk_lw_thread *lw_thread = arg1;

struct spdk_reactor *reactor;

// 消息傳遞到對應的reactor后將該thread加入到reactor中

reactor = spdk_reactor_get(spdk_env_get_current_core());

TAILQ_INSERT_TAIL(&reactor-》threads, lw_thread, link);

}

在SPDK_REACTOR_STATE_RUNNING后,此時所有reactor就進入了輪詢狀態。_spdk_reactor_run函數為線程提供了輪詢方法:

static int _spdk_reactor_run(void *arg) {

while (1) {

// 處理reactor上的event消息,消息會在之后講到

_spdk_event_queue_run_batch(reactor);

// 每一個reactor上注冊的thread進行遍歷并且處理poller事件

TAILQ_FOREACH_SAFE(lw_thread, &reactor-》threads, link, tmp) {

rc = spdk_thread_poll(thread, 0, now);

}

// 檢查reactor的狀態

if (g_reactor_state != SPDK_REACTOR_STATE_RUNNING) {

break;

}

}

}

而當spdk app被調用spdk_app_stop方法后將會相應的通知每一個reactor調用spdk_reactors_stop方法,將g_reactor_state賦值為SPDK_REACTOR_STATE_EXITING,即開始退出了。回到_spdk_reactor_run函數中,輪詢將會被跳出,并且執行銷毀線程的代碼。

static int _spdk_reactor_run(void *arg) {

…。. // 輪詢

TAILQ_FOREACH_SAFE(lw_thread, &reactor-》threads, link, tmp) {

thread = spdk_thread_get_from_ctx(lw_thread);

TAILQ_REMOVE(&reactor-》threads, lw_thread, link);

spdk_set_thread(thread);

spdk_thread_exit(thread);

spdk_thread_destroy(thread);

}

}

在這之后,主線程的_spdk_reactor_run會返回到spdk_reactors_start中,并將g_reactor_state賦值為SPDK_REACTOR_STATE_SHUTDOWN,返回到spdk_app_start中等待應用退出。

最后,總結一下reactors和CPU core以及spdk thread關系應該如圖1所示

fbdba11c-8ecb-11eb-8b86-12bb97331649.png

圖1 CPU cores、reactors和thread關系圖

Reactor生命周期流程圖則如圖2所示

fc609bb0-8ecb-11eb-8b86-12bb97331649.png

圖2 reactor生命周期流程圖

1.2 thread

當Reactors進行輪詢時,除了處理自己的事件消息之外,還會調用注冊在該reactor下面的每一個線程進行輪詢。不過通常一個reactor只有一個thread,在spdk應用中,更多的是注冊多個poller而不是注冊多個thread。具體的輪詢方法為:

Int spdk_thread_poll(struct spdk_thread *thread, uint32_t max_msgs, uint64_t now) {

// 首先先處理ring傳遞過來的消息

msg_count = _spdk_msg_queue_run_batch(thread, max_msgs);

// 調用非定時poller中的方法

TAILQ_FOREACH_REVERSE_SAFE(poller, &thread-》active_pollers,

active_pollers_head, tailq, tmp) {

// 調用poller注冊的方法之前,會對poller狀態檢測且轉換

if (poller-》state == SPDK_POLLER_STATE_UNREGISTERED) {

TAILQ_REMOVE(&thread-》active_pollers, poller, tailq);

free(poller);

continue;

}

poller-》state = SPDK_POLLER_STATE_RUNNING;

// 調用poller注冊的方法

poller_rc = poller-》fn(poller-》arg);

// poller轉換狀態

poller-》state = SPDK_POLLER_STATE_WAITING;

}

// 調用定時poller中的方法

TAILQ_FOREACH_SAFE(poller, &thread-》timer_pollers, tailq, tmp) {

// 類似非定時poller過程,不過會檢查是否到了預定的時間

if (now 《 poller-》next_run_tick) break;

}

// 最后統計時間

}

Io_device 和 io_channel在thread中也是非常重要的概念。它們的實現都在thread.c中,io_device是設備的抽象,io_channel是對該設備通道的抽象。一個線程可以創建多個io_channel 。 io_channel只能和一個io_device綁定,并且這個io_channel是別的線程使用不了的。

fd1f8890-8ecb-11eb-8b86-12bb97331649.png

圖 3 io_device、io_channel和線程關系圖

Io_device結構

struct io_device {

void *io_device; // 抽象的device指針

char name[SPDK_MAX_DEVICE_NAME_LEN + 1]; // 名字

spdk_io_channel_create_cb create_cb; // io_channel創建的回調函數

spdk_io_channel_destroy_cb destroy_cb; // io_channel銷毀的回調函數

spdk_io_device_unregister_cb unregister_cb; // io_device解綁的回調函數

struct spdk_thread *unregister_thread; // 不使用該device線程

uint32_t ctx_size; // ctx的大小,將會傳給io_channel處理

uint32_t for_each_count; // io_channel的數量

TAILQ_ENTRY(io_device) tailq; // device隊列頭

uint32_t refcnt; // 計數器

bool unregistered; // 是否該device被注冊

};

可以看到,io_device實際上只提供了一些自身io_device的操作和io_channel相關的方法,具體的io_device實體其實是那個名字叫io_device的void指針。因為thread中的io_device只提供了thread這一層接口,具體的io操作每一個設備很難被抽象出來,所以這一層的接口只負責管理io_channel的創建、銷毀和綁定等。

Io_channel的結構

struct spdk_io_channel {

struct spdk_thread *thread; // 綁定的線程

struct io_device *dev; // 綁定的io_device

uint32_t ref; // io_channel引用計數

uint32_t destroy_ref; // destroy前被引用的次數

TAILQ_ENTRY(spdk_io_channel) tailq; // io_channel 隊列頭

spdk_io_channel_destroy_cb destroy_cb; // io_channel銷毀的回調函數

};

雖然io_channel看起來是很簡單的結構體,實際上在創建一個io_device的時候,會要求使用者傳入一個io_channel_ctx的大小作為調用的參數,而在給io_channel分配內存的時候,除了分配本身io_channel結構體的大小外,還會額外分配一個io_channel_ctx的大小,這個context可以理解成一個void指針,當用戶在使用io_channel的時候,實際上還是通過context的部分去訪問io_device。

P2

NVMe-oF實例

nvmf_tgt 是spdk中一個重要的模塊,這里詳細的寫一下它作為一個target實例是如何使用thread、io_device以及io_channel的。

在spdk應用剛啟動的時候,reactor模塊就會自動加載起來,然后在加載nvmf subsystem的時候,會調用spdk_nvmf_subsystem_init(lib/event/subsystems/nvmf/nvmf_tgt.c)方法,nvmf_tgt其實也是有生命周期,并且有一個狀態機去管理它的生命周期。

enum nvmf_tgt_state {

NVMF_TGT_INIT_NONE = 0, // 最初的狀態

NVMF_TGT_INIT_PARSE_CONFIG, // 解析配置文件

NVMF_TGT_INIT_CREATE_POLL_GROUPS, // 創建poll groups

NVMF_TGT_INIT_START_SUBSYSTEMS, // 啟動subsystem

NVMF_TGT_INIT_START_ACCEPTOR, // 開始接收

NVMF_TGT_RUNNING, // running

NVMF_TGT_FINI_STOP_SUBSYSTEMS,

NVMF_TGT_FINI_DESTROY_POLL_GROUPS,

NVMF_TGT_FINI_STOP_ACCEPTOR,

NVMF_TGT_FINI_FREE_RESOURCES,

NVMF_TGT_STOPPED,

NVMF_TGT_ERROR,

};

首先在NVMF_TGT_INIT_PARSE_CONFIG狀態中,nvmf_tgt會去解析啟動時傳入的配置文件,當解析了[nvmf]這個label后,會調用spdk_nvmf_tgt_create這個方法,這個方法將初始化了全局的g_nvmf_tgt變量,同時也將tgt注冊成了一個io_device。

spdk_io_device_register(tgt,

spdk_nvmf_tgt_create_poll_group,

spdk_nvmf_tgt_destroy_poll_group,

sizeof(struct spdk_nvmf_poll_group),

“nvmf_tgt”);

spdk_nvmf_tgt_create_poll_group和spdk_nvmf_tgt_destroy_poll_group是io_channel創建和銷毀的回調方法。第三個參數是io_channel_ctx的size,既然這里傳入了spdk_nvmf_poll_group的大小,那么很明顯說明在nvmf中io_channel_ctx對象就是spdk_nvmf_poll_group。

當config文件解析完了之后,nvmf_tgt狀態到了NVMF_TGT_INIT_CREATE_POLL_GROUPS,這個狀態下會為每一個線程都創建相應的poll group。

spdk_for_each_thread(nvmf_tgt_create_poll_group,

NULL,

nvmf_tgt_create_poll_group_done);

static void nvmf_tgt_create_poll_group(void *ctx)

{

struct nvmf_tgt_poll_group *pg;

…。

pg-》thread = spdk_get_thread();

pg-》group = spdk_nvmf_poll_group_create(g_spdk_nvmf_tgt);

…。

}

再看spdk_nvmf_poll_group_create中,

struct spdk_nvmf_poll_group * spdk_nvmf_poll_group_create(struct spdk_nvmf_tgt *tgt)

{

struct spdk_io_channel *ch;

ch = spdk_get_io_channel(tgt);

…。

return spdk_io_channel_get_ctx(ch);

}

在spdk_get_io_channel中,會先去檢查傳入的io_device是不是已經注冊好了的,如果已經注冊了,將會創建一個新的io_channel返回,創建的過程會回調在注冊io_device時注冊的io_channel創建方法(即方法spdk_nvmf_tgt_create_poll_group)。

static int spdk_nvmf_tgt_create_poll_group(void *io_device, void *ctx_buf)

{

…。. // 初始化transport 、nvmf subsystem等

// 注冊一個poller

group-》poller = spdk_poller_register(spdk_nvmf_poll_group_poll, group, 0);

group-》thread = spdk_get_thread();

return 0;

}

在spdk_nvmf_poll_group_poll中,因為spdk_nvmf_poll_group對象中有transport的poll group,所以它會調用對應的transport的poll_group_poll方法,比如rdma的poll_group_poll就會輪詢rdma注冊的poller處理每個在相應的qpair來的請求,進入rdma的狀態機將請求處理好。

然后這個狀態就結束了,之后再初始化好了nvmf subsystem相關的東西之后,到了狀態NVMF_TGT_INIT_START_ACCEPTOR。在這個狀態中,只注冊了一個poller。

g_acceptor_poller = spdk_poller_register(acceptor_poll, g_spdk_nvmf_tgt,

g_spdk_nvmf_tgt_conf-》acceptor_poll_rate);

這個poller調用的transport的方法,不斷的監聽是不是有新的fd連接進來,如果有就調用new_qpair的回調。

P3

總結

spdk thread 模型是spdk無鎖化的基礎,在一個線程中,當分配一個任務后,一直會運行到任務結束為止,這確保了不需要進行線程之間的切換而帶來額外的損耗。同時,高效的spdk ring提供了不同線程之間的消息傳遞,這就使得任務結束的結果可以高效的傳遞給別的處理線程。而io_device和io_channel的設計保證了資源的抽象訪問以及獨立的路徑不去爭搶資源池,并且塊設備由于是對塊進行操作的所以也十分適合抽象成io_device。正是因為以上幾點才讓spdk線程模型能夠達到無鎖化且為多個target提供了基礎線程框架的支持。

原文標題:SPDK線程模型解析

文章出處:【微信公眾號:FPGA之家】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

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

    關注

    68

    文章

    19811

    瀏覽量

    233600
  • 線程
    +關注

    關注

    0

    文章

    507

    瀏覽量

    20080

原文標題:SPDK線程模型解析

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦
    熱點推薦

    Thread標準認證概述

    本篇知識庫文章概述了開發人員如何將其Thread物聯網設備進行Thread Group認證所需的步驟,并重點介紹使用Silicon Labs(芯科科技)的EFR32無線射頻器件的相關流程。
    的頭像 發表于 06-04 10:10 ?95次閱讀
    <b class='flag-5'>Thread</b>標準認證概述

    RT-Thread 上如何實現 SLAAC?

    大佬們,本菜鳥有一些網絡上的問題需要幫助: RT-Thread 上如何實現 SLAAC(無狀態地址自動分配),給連接到我的板子的設備分配 IPv6 地址; RT-Thread 如何發送以太網報文,要求從報文頭開始都是我自己組,
    發表于 05-27 07:21

    用于 Zigbee?/Thread/藍牙?信號應用的 2.4 GHz 前端模塊 skyworksinc

    電子發燒友網為你提供()用于 Zigbee?/Thread/藍牙?信號應用的 2.4 GHz 前端模塊相關產品參數、數據手冊,更有用于 Zigbee?/Thread/藍牙?信號應用的 2.4 GHz
    發表于 05-09 18:31
    用于 Zigbee?/<b class='flag-5'>Thread</b>/藍牙?信號應用的 2.4 GHz 前端模塊 skyworksinc

    如何基于Kahn處理網絡定義AI引擎圖形編程模型

    本白皮書探討了如何基于 Kahn 處理網絡( KPN )定義 AI 引擎圖形編程模型。KPN 模型有助于實現數據流并行化,進而提高系統的整體性能。
    的頭像 發表于 04-17 11:31 ?260次閱讀
    如何基于Kahn處理網絡定義AI引擎圖形編程<b class='flag-5'>模型</b>

    用于 Zigbee?/Thread/藍牙?應用的 2.4 GHz 前端模塊 skyworksinc

    電子發燒友網為你提供()用于 Zigbee?/Thread/藍牙?應用的 2.4 GHz 前端模塊相關產品參數、數據手冊,更有用于 Zigbee?/Thread/藍牙?應用的 2.4 GHz 前端
    發表于 04-11 18:30
    用于 Zigbee?/<b class='flag-5'>Thread</b>/藍牙?應用的 2.4 GHz 前端模塊 skyworksinc

    通用Matter over thread 模組

    PTR5415是一款通用Matter over thread 模組,它支持接入蘋果等matter over thread 生態,實現基于matter over thread的連接和控制
    發表于 03-26 14:42

    移遠通信AI玩具整體解決方案全面升級:融合火山引擎RTC大模型,打造實時交互新體驗

    2月20日,全球領先的物聯網整體解決方案供應商移遠通信宣布,其AI玩具整體解決方案已實現全面的完善和升級。該方案深度融合火山引擎AI大模型能力,集無線通信模組、音頻算法、物聯網平臺和收
    發表于 02-21 09:50 ?347次閱讀
    移遠通信AI玩具<b class='flag-5'>整體</b>解決方案全面升級:融合火山引擎RTC大<b class='flag-5'>模型</b>,打造實時交互新體驗

    移遠通信AI玩具整體解決方案全面升級:融合火山引擎RTC大模型,打造實時交互新體驗

    2月20日,移遠通信宣布,其AI玩具整體解決方案已實現全面的完善和升級。該方案深度融合火山引擎AI大模型能力,集無線通信模組、音頻算法、物聯網平臺和收費管理平臺于一體,可為玩具的智能化升級提供從硬件
    的頭像 發表于 02-20 19:06 ?457次閱讀
    移遠通信AI玩具<b class='flag-5'>整體</b>解決方案全面升級:融合火山引擎RTC大<b class='flag-5'>模型</b>,打造實時交互新體驗

    Deepseek上單片機?RT-Thread上跑通大語言模型

    前言單片機也能聊天?RT-Thread上跑通大語言模型在RT-Thread論壇上忽然看到了單片機和大模型對話的文章,想著春節期間看到大語言模型
    的頭像 發表于 02-07 18:59 ?1327次閱讀
    Deepseek上單片機?RT-<b class='flag-5'>Thread</b>上跑通大語言<b class='flag-5'>模型</b>

    2024年Thread的重要亮點

    Thread Group近期通過本篇博文來總結2024年取得的驚人進步和成就。目前,Thread會員基礎已近 200 家公司,從去年1月份參加CES 2024,到7月份慶祝成立 10 周年,再到
    的頭像 發表于 01-14 09:30 ?574次閱讀

    基于NXP MCXA153 MCU實現RT-Thread的MTD NOR Flash驅動

    在嵌入式系統中,片上Flash存儲器是一個關鍵組件,用于存儲程序代碼和關鍵數據。本文將詳細介紹如何在NXPMCXA153 MCU上實現RT-Thread的MTD (Memory Technology Device) NOR Flash驅動,以管理128KB的片上Flash
    的頭像 發表于 11-09 14:00 ?1035次閱讀
    基于NXP MCXA153 MCU<b class='flag-5'>實現</b>RT-<b class='flag-5'>Thread</b>的MTD NOR Flash驅動

    基于恩智浦FRDM-MCXA153開發板實現RT-Thread的ADC驅動

    數轉換器(ADC)是現代嵌入式系統中不可或缺的組件,它能將連續的模擬信號轉換為離散的數字信號。本文將深入探討如何在NXP的FRDM-MCXA153開發板上實現和使用RT-Thread的ADC驅動,為開發者提供一個全面而實用的指南。
    的頭像 發表于 11-01 12:29 ?1300次閱讀
    基于恩智浦FRDM-MCXA153開發板<b class='flag-5'>實現</b>RT-<b class='flag-5'>Thread</b>的ADC驅動

    開源共生 商業共贏 | RT-Thread 2024開發者大會報名啟動!

    開發者大會將以“開源共生,商業共贏”為主題,將于2024年12月21日全天在上海臨港中心舉行。開源RT-Thread如何實現商業化?這一直是外界對RT-Thread
    的頭像 發表于 10-29 08:06 ?866次閱讀
    開源共生 商業共贏 | RT-<b class='flag-5'>Thread</b> 2024開發者大會報名啟動!

    Nordic-RT-Thread5.1.0移植筆記

    Nordic-RT-Thread5.1.0移植筆記
    的頭像 發表于 10-16 08:09 ?1148次閱讀
    Nordic-RT-<b class='flag-5'>Thread</b>5.1.0移植筆記

    中興通訊推出星云通信大模型

    在2024MWC上海展期間,中興通訊舉辦了星云通信大模型發布會,正式推出星云通信大模型系列化產品,涵蓋星云通信大模型、Agent工廠和系列化應用產品。通過從模型到平臺再到應用,提供煉好
    的頭像 發表于 09-14 14:18 ?913次閱讀