基础代码NVMe模块的实例helloworld代码

简介: 基础代码NVMe模块的实例helloworld代码


干货直接上代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <spdk/nvme.h>
#define NUM_BLOCKS 1024
#define BLOCK_SIZE 4096
#define QUEUE_DEPTH 128
static struct spdk_nvme_ctrlr *g_ctrlr = NULL;
static struct spdk_nvme_ns *g_ns = NULL;
static void
init_nvme(void)
{
    struct spdk_nvme_transport_id trid = {0};
    struct spdk_nvme_ctrlr_opts opts = {0};
    struct spdk_nvme_ns_list *ns_list = NULL;
    struct spdk_nvme_ns *ns = NULL;
    int rc;
    /* Set up NVMe controller options */
    opts.use_cmb_sqs = true;
    opts.keep_alive_timeout_ms = 0;
    /* Set up NVMe transport identifier */
    trid.trtype = SPDK_NVME_TRANSPORT_PCIE;
    trid.adrfam = SPDK_NVMF_ADRFAM_IPV4;
    trid.traddr = "0000:04:00.0";
    /* Probe for NVMe controllers */
    rc = spdk_nvme_probe(&trid, NULL, NULL, NULL, &ns_list);
    if (rc != 0) {
        fprintf(stderr, "Failed to probe for NVMe controllers: %!s(MISSING)\n", spdk_strerror(-rc));
        exit(1);
    }
    /* Connect to the first NVMe controller */
    g_ctrlr = spdk_nvme_connect(&trid, &opts, 0);
    if (g_ctrlr == NULL) {
        fprintf(stderr, "Failed to connect to NVMe controller\n");
        exit(1);
    }
    /* Get the first namespace */
    ns = spdk_nvme_ns_list_first(ns_list);
    if (ns == NULL) {
        fprintf(stderr, "No NVMe namespaces found\n");
        exit(1);
    }
    /* Connect to the namespace */
    g_ns = spdk_nvme_ctrlr_get_ns(g_ctrlr, spdk_nvme_ns_get_id(ns));
    if (g_ns == NULL) {
        fprintf(stderr, "Failed to connect to NVMe namespace\n");
        exit(1);
    }
}
static void
io_complete(void *cb_arg, const struct spdk_nvme_cpl *completion)
{
    printf("I/O completed\n");
}
int main(int argc, char **argv)
{
    int i, j, rc;
    void *buf[NUM_BLOCKS];
    /* Initialize SPDK and NVMe */
    spdk_env_init(NULL);
    init_nvme();
    /* Allocate memory buffers */
    for (i = 0; i < NUM_BLOCKS; i++) {
        buf[i] = spdk_dma_zmalloc(BLOCK_SIZE, BLOCK_SIZE, NULL);
        if (buf[i] == NULL) {
            fprintf(stderr, "Failed to allocate memory buffer\n");
            exit(1);
        }
    }
    /* Submit I/O requests */
    for (i = 0; i < NUM_BLOCKS; i += QUEUE_DEPTH) {
        struct spdk_nvme_qpair *qpair = spdk_nvme_ctrlr_alloc_io_qpair(g_ctrlr, NULL, 0);
        if (qpair == NULL) {
            fprintf(stderr, "Failed to allocate I/O queue pair\n");
            exit(1);
        }
        for (j = 0; j < QUEUE_DEPTH && i + j < NUM_BLOCKS; j++) {
            rc = spdk_nvme_ns_cmd_write(g_ns, qpair, buf[i + j], i + j, 1, io_complete, NULL, 0);
            if (rc != 0) {
                fprintf(stderr, "Failed to submit write I/O request: %!s(MISSING)\n", spdk_strerror(-rc));
                exit(1);
            }
        }
        spdk_nvme_qpair_process_completions(qpair, -1);
        spdk_nvme_ctrlr_free_io_qpair(g_ctrlr, qpair);
    }
    /* Free memory buffers */
    for (i = 0; i < NUM_BLOCKS; i++) {
        spdk_dma_free(buf[i]);
    }
    /* Disconnect from NVMe namespace and controller */
    spdk_nvme_detach(g_ctrlr);
    spdk_env_cleanup();
    return 0;
}

这个代码的作用是向NVMe SSD写入一些数据。它的执行流程如下:

  1. 初始化SPDK环境。
  2. 初始化NVMe控制器。
  3. 分配一些内存缓冲区。
  4. 提交I/O请求,每次提交QUEUE_DEPTH个写请求,其中每个请求写入一个BLOCK_SIZE大小的数据块。
  5. 等待I/O请求完成。
  6. 释放内存缓冲区。
  7. 断开与NVMe SSD的连接。
  8. 清理SPDK环境。
  9. 在这个代码中,我们使用了以下的NVMe相关函数:
  • spdk_nvme_probe:探测NVMe控制器。
  • spdk_nvme_connect:连接到NVMe控制器。
  • spdk_nvme_ctrlr_get_ns:连接到NVMe命名空间。
  • spdk_dma_zmalloc 和 spdk_dma_free:分配和释放DMA内存
  • spdk_nvme_ctrlr_alloc_io_qpair 和 spdk_nvme_ctrlr_free_io_qpair:分配和释放I/O队列对。
  • spdk_nvme_ns_cmd_write:提交写I/O请求。
  • spdk_nvme_qpair_process_completions:处理I/O请求完成事件。
  • spdk_nvme_detach:断开与NVMe SSD的连接。
    这个代码演示了如何使用SPDK的NVMe模块向NVMe SSD写入数据。它使用了SPDK的异步I/O架构,可以在保持低延迟的同时实现高吞吐量。在实际的应用程序中,我们可以根据需要修改这个代码,比如读取数据、使用多个队列、使用多个命名空间等。

推荐一个零声学院免费公开课程,个人觉得老师讲得不错,分享给大家:Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK等技术内容,立即学习

相关文章
|
3月前
|
消息中间件 JavaScript 中间件
函数计算产品使用问题之WebIDE编写的Node.js代码是否会自动进行打包部署
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
5月前
|
数据采集 人工智能 测试技术
模块化重构LLaVA,替换组件只需添加1-2个文件,开源TinyLLaVA Factory来了
【6月更文挑战第5天】TinyLLaVA Factory是新推出的开源模块化代码库,简化小规模多模态模型的设计与训练,采用工厂模式,允许用户通过添加或替换少量文件轻松重构模型组件,提高开发效率。该库支持定制LMMs并提供训练配方,通过模块化和开源促进社区合作,已实验证实在复现性能上与原始模型相当。尽管面临兼容性等挑战,但其前瞻性的技术路线图确保了其在AI领域的相关性和活力。论文链接:[arxiv.org/abs/2405.11788](https://arxiv.org/abs/2405.11788)
78 0
|
前端开发
使用concurrently模块-同时启动react项目和mock模拟接口
使用concurrently模块-同时启动react项目和mock模拟接口
83 0
|
6月前
|
存储 固态存储 API
spdk关于nvme模块的实例helloword代码
spdk关于nvme模块的实例helloword代码
|
6月前
|
JavaScript
Vue项目常见的文档导入调接口
Vue项目常见的文档导入调接口
|
6月前
|
算法 关系型数据库 编译器
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目(一)
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目
196 0
|
6月前
|
C++
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目(二)
[项目配置] 配置Qt函数库和ui界面库的封装并调用的项目
105 0
|
JSON 数据格式 Python
Python基础 模块化编程(模块的导入) 模块化编程 模块以主程序的方式运行 包和目录 第三方库的安装和导入方法
python基础知识模块,模块化编程,模块的创建和导入 python基础,模块的创建和导入,让模块以主程序的方式运行,python中的包和目录的区别和创建。模块导入另一个包的模块的方法,导入带有包的模块时的注意事项,常见的内置模块。 第三方模块的安装和导入的方法
Python基础 模块化编程(模块的导入)   模块化编程 模块以主程序的方式运行 包和目录 第三方库的安装和导入方法
|
数据库
配置模块案例工程代码讲解——基于 NestJS 7.x
配置模块案例工程代码讲解——基于 NestJS 7.x
182 0
|
存储 Java Unix
浅析Python模块的引入和调用
浅析Python模块的引入和调用
136 0
浅析Python模块的引入和调用