/*- * BSD LICENSE * * Copyright (c) Intel Corporation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * Neither the name of Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /** \file * NVMe driver public API */ #ifndef SPDK_NVME_H #define SPDK_NVME_H #include "spdk/stdinc.h" #ifdef __cplusplus extern "C" { #endif #include "spdk/env.h" #include "spdk/nvme_spec.h" #include "spdk/nvmf_spec.h" #define SPDK_NVME_DEFAULT_RETRY_COUNT (4) extern int32_t spdk_nvme_retry_count; /** \brief Opaque handle to a controller. Returned by \ref spdk_nvme_probe()'s attach_cb. */ struct spdk_nvme_ctrlr; /** * \brief NVMe controller initialization options. * * A pointer to this structure will be provided for each probe callback from spdk_nvme_probe() to * allow the user to request non-default options, and the actual options enabled on the controller * will be provided during the attach callback. */ struct spdk_nvme_ctrlr_opts { /** * Number of I/O queues to request (used to set Number of Queues feature) */ uint32_t num_io_queues; /** * Enable submission queue in controller memory buffer */ bool use_cmb_sqs; /** * Type of arbitration mechanism */ enum spdk_nvme_cc_ams arb_mechanism; /** * Keep alive timeout in milliseconds (0 = disabled). * * The NVMe library will set the Keep Alive Timer feature to this value and automatically * send Keep Alive commands as needed. The library user must call * spdk_nvme_ctrlr_process_admin_completions() periodically to ensure Keep Alive commands * are sent. */ uint32_t keep_alive_timeout_ms; /** * Specify the retry number when there is issue with the transport */ int transport_retry_count; /** * The queue depth of each NVMe I/O queue. */ uint32_t io_queue_size; /** * The host NQN to use when connecting to NVMe over Fabrics controllers. * * Unused for local PCIe-attached NVMe devices. */ char hostnqn[SPDK_NVMF_NQN_MAX_LEN + 1]; /** * The number of requests to allocate for each NVMe I/O queue. * * This should be at least as large as io_queue_size. * * A single I/O may allocate more than one request, since splitting may be necessary to * conform to the device's maximum transfer size, PRP list compatibility requirements, * or driver-assisted striping. */ uint32_t io_queue_requests; }; /** * NVMe library transports * * NOTE: These are mapped directly to the NVMe over Fabrics TRTYPE values, except for PCIe, * which is a special case since NVMe over Fabrics does not define a TRTYPE for local PCIe. * * Currently, this uses 256 for PCIe which is intentionally outside of the 8-bit range of TRTYPE. * If the NVMe-oF specification ever defines a PCIe TRTYPE, this should be updated. */ enum spdk_nvme_transport_type { /** * PCIe Transport (locally attached devices) */ SPDK_NVME_TRANSPORT_PCIE = 256, /** * RDMA Transport (RoCE, iWARP, etc.) */ SPDK_NVME_TRANSPORT_RDMA = SPDK_NVMF_TRTYPE_RDMA, }; /** * NVMe transport identifier. * * This identifies a unique endpoint on an NVMe fabric. * * A string representation of a transport ID may be converted to this type using * spdk_nvme_transport_id_parse(). */ struct spdk_nvme_transport_id { /** * NVMe transport type. */ enum spdk_nvme_transport_type trtype; /** * Address family of the transport address. * * For PCIe, this value is ignored. */ enum spdk_nvmf_adrfam adrfam; /** * Transport address of the NVMe-oF endpoint. For transports which use IP * addressing (e.g. RDMA), this should be an IP address. For PCIe, this * can either be a zero length string (the whole bus) or a PCI address * in the format DDDD:BB:DD.FF or DDDD.BB.DD.FF */ char traddr[SPDK_NVMF_TRADDR_MAX_LEN + 1]; /** * Transport service id of the NVMe-oF endpoint. For transports which use * IP addressing (e.g. RDMA), this field shoud be the port number. For PCIe, * this is always a zero length string. */ char trsvcid[SPDK_NVMF_TRSVCID_MAX_LEN + 1]; /** * Subsystem NQN of the NVMe over Fabrics endpoint. May be a zero length string. */ char subnqn[SPDK_NVMF_NQN_MAX_LEN + 1]; }; /** * Parse the string representation of a transport ID. * * \param trid Output transport ID structure (must be allocated and initialized by caller). * \param str Input string representation of a transport ID to parse. * \return 0 if parsing was successful and trid is filled out, or negated errno values on failure. * * str must be a zero-terminated C string containing one or more key:value pairs separated by * whitespace. * * Key | Value * ------------ | ----- * trtype | Transport type (e.g. PCIe, RDMA) * adrfam | Address family (e.g. IPv4, IPv6) * traddr | Transport address (e.g. 0000:04:00.0 for PCIe or 192.168.100.8 for RDMA) * trsvcid | Transport service identifier (e.g. 4420) * subnqn | Subsystem NQN * * Unspecified fields of trid are left unmodified, so the caller must initialize trid (for example, * memset() to 0) before calling this function. */ int spdk_nvme_transport_id_parse(struct spdk_nvme_transport_id *trid, const char *str); /** * Parse the string representation of a transport ID tranport type. * * \param trtype Output transport type (allocated by caller). * \param str Input string representation of transport type (e.g. "PCIe", "RDMA") * \return 0 if parsing was successful and trtype is filled out, or negated errno values on failure. */ int spdk_nvme_transport_id_parse_trtype(enum spdk_nvme_transport_type *trtype, const char *str); /** * Look up the string representation of a transport ID transport type. * * \param trtype Transport type to convert. * \return Static string constant describing trtype, or NULL if trtype not found. */ const char *spdk_nvme_transport_id_trtype_str(enum spdk_nvme_transport_type trtype); /** * Look up the string representation of a transport ID address family. * * \param adrfam Address family to convert. * \return Static string constant describing adrfam, or NULL if adrmfam not found. */ const char *spdk_nvme_transport_id_adrfam_str(enum spdk_nvmf_adrfam adrfam); /** * Parse the string representation of a tranport ID address family. * * \param adrfam Output address family (allocated by caller). * \param str Input string representation of address family (e.g. "IPv4", "IPv6") * \return 0 if parsing was successful and adrfam is filled out, or negated errno values on failure. */ int spdk_nvme_transport_id_parse_adrfam(enum spdk_nvmf_adrfam *adrfam, const char *str); /** * Compare two transport IDs. * * \param trid1 First transport ID to compare. * \param trid2 Second transport ID to compare. * * \return 0 if trid1 == trid2, less than 0 if trid1 < trid2, greater than 0 if trid1 > trid2. * * The result of this function may be used to sort transport IDs in a consistent order; however, * the comparison result is not guaranteed to be consistent across library versions. * * This function uses a case-insensitive comparison for string fields, but it does not otherwise * normalize the transport ID. It is the caller's responsibility to provide the transport IDs in * a consistent format. */ int spdk_nvme_transport_id_compare(const struct spdk_nvme_transport_id *trid1, const struct spdk_nvme_transport_id *trid2); /** * Determine whether the NVMe library can handle a specific NVMe over Fabrics transport type. * * \param trtype NVMe over Fabrics transport type to check. * * \return true if trtype is supported or false if it is not supported. */ bool spdk_nvme_transport_available(enum spdk_nvme_transport_type trtype); /** * Callback for spdk_nvme_probe() enumeration. * * \param opts NVMe controller initialization options. This structure will be populated with the * default values on entry, and the user callback may update any options to request a different * value. The controller may not support all requested parameters, so the final values will be * provided during the attach callback. * \return true to attach to this device. */ typedef bool (*spdk_nvme_probe_cb)(void *cb_ctx, const struct spdk_nvme_transport_id *trid, struct spdk_nvme_ctrlr_opts *opts); /** * Callback for spdk_nvme_probe() to report a device that has been attached to the userspace NVMe driver. * * \param opts NVMe controller initialization options that were actually used. Options may differ * from the requested options from the probe call depending on what the controller supports. */ typedef void (*spdk_nvme_attach_cb)(void *cb_ctx, const struct spdk_nvme_transport_id *trid, struct spdk_nvme_ctrlr *ctrlr, const struct spdk_nvme_ctrlr_opts *opts); /** * Callback for spdk_nvme_probe() to report that a device attached to the userspace NVMe driver * has been removed from the system. * * The controller will remain in a failed state (any new I/O submitted will fail). * * The controller must be detached from the userspace driver by calling spdk_nvme_detach() * once the controller is no longer in use. It is up to the library user to ensure that * no other threads are using the controller before calling spdk_nvme_detach(). * * \param ctrlr NVMe controller instance that was removed. */ typedef void (*spdk_nvme_remove_cb)(void *cb_ctx, struct spdk_nvme_ctrlr *ctrlr); /** * \brief Enumerate the bus indicated by the transport ID and attach the userspace NVMe driver * to each device found if desired. * * \param trid The transport ID indicating which bus to enumerate. If the trtype is PCIe or trid is NULL, * this will scan the local PCIe bus. If the trtype is RDMA, the traddr and trsvcid must point at the * location of an NVMe-oF discovery service. * \param cb_ctx Opaque value which will be passed back in cb_ctx parameter of the callbacks. * \param probe_cb will be called once per NVMe device found in the system. * \param attach_cb will be called for devices for which probe_cb returned true once that NVMe * controller has been attached to the userspace driver. * \param remove_cb will be called for devices that were attached in a previous spdk_nvme_probe() * call but are no longer attached to the system. Optional; specify NULL if removal notices are not * desired. * * This function is not thread safe and should only be called from one thread at a time while no * other threads are actively using any NVMe devices. * * If called from a secondary process, only devices that have been attached to the userspace driver * in the primary process will be probed. * * If called more than once, only devices that are not already attached to the SPDK NVMe driver * will be reported. * * To stop using the the controller and release its associated resources, * call \ref spdk_nvme_detach with the spdk_nvme_ctrlr instance returned by this function. */ int spdk_nvme_probe(const struct spdk_nvme_transport_id *trid, void *cb_ctx, spdk_nvme_probe_cb probe_cb, spdk_nvme_attach_cb attach_cb, spdk_nvme_remove_cb remove_cb); /** * \brief Detaches specified device returned by \ref spdk_nvme_probe()'s attach_cb from the NVMe driver. * * On success, the spdk_nvme_ctrlr handle is no longer valid. * * This function should be called from a single thread while no other threads * are actively using the NVMe device. * */ int spdk_nvme_detach(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Perform a full hardware reset of the NVMe controller. * * This function should be called from a single thread while no other threads * are actively using the NVMe device. * * Any pointers returned from spdk_nvme_ctrlr_get_ns() and spdk_nvme_ns_get_data() may be invalidated * by calling this function. The number of namespaces as returned by spdk_nvme_ctrlr_get_num_ns() may * also change. */ int spdk_nvme_ctrlr_reset(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Get the identify controller data as defined by the NVMe specification. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * */ const struct spdk_nvme_ctrlr_data *spdk_nvme_ctrlr_get_data(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Get the NVMe controller CSTS (Status) register. */ union spdk_nvme_csts_register spdk_nvme_ctrlr_get_regs_csts(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Get the NVMe controller CAP (Capabilities) register. */ union spdk_nvme_cap_register spdk_nvme_ctrlr_get_regs_cap(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Get the NVMe controller VS (Version) register. */ union spdk_nvme_vs_register spdk_nvme_ctrlr_get_regs_vs(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Get the number of namespaces for the given NVMe controller. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * This is equivalent to calling spdk_nvme_ctrlr_get_data() to get the * spdk_nvme_ctrlr_data and then reading the nn field. * */ uint32_t spdk_nvme_ctrlr_get_num_ns(struct spdk_nvme_ctrlr *ctrlr); /** * \brief Determine if a particular log page is supported by the given NVMe controller. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * \sa spdk_nvme_ctrlr_cmd_get_log_page() */ bool spdk_nvme_ctrlr_is_log_page_supported(struct spdk_nvme_ctrlr *ctrlr, uint8_t log_page); /** * \brief Determine if a particular feature is supported by the given NVMe controller. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * \sa spdk_nvme_ctrlr_cmd_get_feature() */ bool spdk_nvme_ctrlr_is_feature_supported(struct spdk_nvme_ctrlr *ctrlr, uint8_t feature_code); /** * Signature for callback function invoked when a command is completed. * * The spdk_nvme_cpl parameter contains the completion status. */ typedef void (*spdk_nvme_cmd_cb)(void *, const struct spdk_nvme_cpl *); /** * Signature for callback function invoked when an asynchronous error * request command is completed. * * The aer_cb_arg parameter is set to the context specified by * spdk_nvme_register_aer_callback(). * The spdk_nvme_cpl parameter contains the completion status of the * asynchronous event request that was completed. */ typedef void (*spdk_nvme_aer_cb)(void *aer_cb_arg, const struct spdk_nvme_cpl *); void spdk_nvme_ctrlr_register_aer_callback(struct spdk_nvme_ctrlr *ctrlr, spdk_nvme_aer_cb aer_cb_fn, void *aer_cb_arg); /** * \brief Opaque handle to a queue pair. * * I/O queue pairs may be allocated using spdk_nvme_ctrlr_alloc_io_qpair(). */ struct spdk_nvme_qpair; /** * Signature for the callback function invoked when a timeout is * detected on a request. * For timeouts detected on the admin queue pair, the qpair returned * here will be NULL. */ typedef void (*spdk_nvme_timeout_cb)(void *cb_arg, struct spdk_nvme_ctrlr *ctrlr, struct spdk_nvme_qpair *qpair, uint16_t cid); /** * \brief Register for timeout callback on a controller. * * The application can choose to register for timeout callback or not register * for timeout callback. * * \param ctrlr NVMe controller on which to monitor for timeout. * \param timeout_sec Timeout value in seconds. * \param cb_fn A function pointer that points to the callback function * \param cb_arg Argument to the callback function. */ void spdk_nvme_ctrlr_register_timeout_callback(struct spdk_nvme_ctrlr *ctrlr, uint32_t timeout_sec, spdk_nvme_timeout_cb cb_fn, void *cb_arg); /** * \brief Allocate an I/O queue pair (submission and completion queue). * * Each queue pair should only be used from a single thread at a time (mutual exclusion must be * enforced by the user). * * \param ctrlr NVMe controller for which to allocate the I/O queue pair. * \param qprio Queue priority for weighted round robin arbitration. If a different arbitration * method is in use, pass 0. */ struct spdk_nvme_qpair *spdk_nvme_ctrlr_alloc_io_qpair(struct spdk_nvme_ctrlr *ctrlr, enum spdk_nvme_qprio qprio); /** * \brief Free an I/O queue pair that was allocated by spdk_nvme_ctrlr_alloc_io_qpair(). */ int spdk_nvme_ctrlr_free_io_qpair(struct spdk_nvme_qpair *qpair); /** * \brief Send the given NVM I/O command to the NVMe controller. * * This is a low level interface for submitting I/O commands directly. Prefer * the spdk_nvme_ns_cmd_* functions instead. The validity of the command will * not be checked! * * When constructing the nvme_command it is not necessary to fill out the PRP * list/SGL or the CID. The driver will handle both of those for you. * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ctrlr_cmd_io_raw(struct spdk_nvme_ctrlr *ctrlr, struct spdk_nvme_qpair *qpair, struct spdk_nvme_cmd *cmd, void *buf, uint32_t len, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Process any outstanding completions for I/O submitted on a queue pair. * * This call is non-blocking, i.e. it only * processes completions that are ready at the time of this function call. It does not * wait for outstanding commands to finish. * * For each completed command, the request's callback function will * be called if specified as non-NULL when the request was submitted. * * \param qpair Queue pair to check for completions. * \param max_completions Limit the number of completions to be processed in one call, or 0 * for unlimited. * * \return Number of completions processed (may be 0) or negative on error. * * \sa spdk_nvme_cmd_cb * * This function may be called at any point while the controller is attached to * the SPDK NVMe driver. * * The caller must ensure that each queue pair is only used from one thread at a time. */ int32_t spdk_nvme_qpair_process_completions(struct spdk_nvme_qpair *qpair, uint32_t max_completions); /** * \brief Send the given admin command to the NVMe controller. * * This is a low level interface for submitting admin commands directly. Prefer * the spdk_nvme_ctrlr_cmd_* functions instead. The validity of the command will * not be checked! * * When constructing the nvme_command it is not necessary to fill out the PRP * list/SGL or the CID. The driver will handle both of those for you. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. */ int spdk_nvme_ctrlr_cmd_admin_raw(struct spdk_nvme_ctrlr *ctrlr, struct spdk_nvme_cmd *cmd, void *buf, uint32_t len, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Process any outstanding completions for admin commands. * * This will process completions for admin commands submitted on any thread. * * This call is non-blocking, i.e. it only processes completions that are ready * at the time of this function call. It does not wait for outstanding commands to * finish. * * \return Number of completions processed (may be 0) or negative on error. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ int32_t spdk_nvme_ctrlr_process_admin_completions(struct spdk_nvme_ctrlr *ctrlr); /** \brief Opaque handle to a namespace. Obtained by calling spdk_nvme_ctrlr_get_ns(). */ struct spdk_nvme_ns; /** * \brief Get a handle to a namespace for the given controller. * * Namespaces are numbered from 1 to the total number of namespaces. There will never * be any gaps in the numbering. The number of namespaces is obtained by calling * spdk_nvme_ctrlr_get_num_ns(). * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ struct spdk_nvme_ns *spdk_nvme_ctrlr_get_ns(struct spdk_nvme_ctrlr *ctrlr, uint32_t ns_id); /** * \brief Get a specific log page from the NVMe controller. * * \param ctrlr NVMe controller to query. * \param log_page The log page identifier. * \param nsid Depending on the log page, this may be 0, a namespace identifier, or SPDK_NVME_GLOBAL_NS_TAG. * \param payload The pointer to the payload buffer. * \param payload_size The size of payload buffer. * \param offset Offset in bytes within the log page to start retrieving log page data. * May only be non-zero if the controller supports extended data for Get Log Page * as reported in the controller data log page attributes. * \param cb_fn Callback function to invoke when the log page has been retrieved. * \param cb_arg Argument to pass to the callback function. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. * * \sa spdk_nvme_ctrlr_is_log_page_supported() */ int spdk_nvme_ctrlr_cmd_get_log_page(struct spdk_nvme_ctrlr *ctrlr, uint8_t log_page, uint32_t nsid, void *payload, uint32_t payload_size, uint64_t offset, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Abort a specific previously-submitted NVMe command. * * \param ctrlr NVMe controller to which the command was submitted. * \param qpair NVMe queue pair to which the command was submitted. * For admin commands, pass NULL for the qpair. * \param cid Command ID of the command to abort. * \param cb_fn Callback function to invoke when the abort has completed. * \param cb_arg Argument to pass to the callback function.\ * * \return 0 if successfully submitted, negated errno value otherwise. * * \sa spdk_nvme_ctrlr_register_timeout_callback() */ int spdk_nvme_ctrlr_cmd_abort(struct spdk_nvme_ctrlr *ctrlr, struct spdk_nvme_qpair *qpair, uint16_t cid, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Set specific feature for the given NVMe controller. * * \param ctrlr NVMe controller to manipulate. * \param feature The feature identifier. * \param cdw11 as defined by the specification for this command. * \param cdw12 as defined by the specification for this command. * \param payload The pointer to the payload buffer. * \param payload_size The size of payload buffer. * \param cb_fn Callback function to invoke when the feature has been set. * \param cb_arg Argument to pass to the callback function. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. * * \sa spdk_nvme_ctrlr_cmd_get_feature() */ int spdk_nvme_ctrlr_cmd_set_feature(struct spdk_nvme_ctrlr *ctrlr, uint8_t feature, uint32_t cdw11, uint32_t cdw12, void *payload, uint32_t payload_size, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Get specific feature from given NVMe controller. * * \param ctrlr NVMe controller to query. * \param feature The feature identifier. * \param cdw11 as defined by the specification for this command. * \param payload The pointer to the payload buffer. * \param payload_size The size of payload buffer. * \param cb_fn Callback function to invoke when the feature has been retrieved. * \param cb_arg Argument to pass to the callback function. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. * * \sa spdk_nvme_ctrlr_cmd_set_feature() */ int spdk_nvme_ctrlr_cmd_get_feature(struct spdk_nvme_ctrlr *ctrlr, uint8_t feature, uint32_t cdw11, void *payload, uint32_t payload_size, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Attach the specified namespace to controllers. * * \param ctrlr NVMe controller to use for command submission. * \param nsid Namespace identifier for namespace to attach. * \param payload The pointer to the controller list. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point after spdk_nvme_attach(). * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. */ int spdk_nvme_ctrlr_attach_ns(struct spdk_nvme_ctrlr *ctrlr, uint32_t nsid, struct spdk_nvme_ctrlr_list *payload); /** * \brief Detach the specified namespace from controllers. * * \param ctrlr NVMe controller to use for command submission. * \param nsid Namespace ID to detach. * \param payload The pointer to the controller list. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point after spdk_nvme_attach(). * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. */ int spdk_nvme_ctrlr_detach_ns(struct spdk_nvme_ctrlr *ctrlr, uint32_t nsid, struct spdk_nvme_ctrlr_list *payload); /** * \brief Create a namespace. * * \param ctrlr NVMe controller to create namespace on. * \param payload The pointer to the NVMe namespace data. * * \return Namespace ID (>= 1) if successfully created, or 0 if the request failed. * * This function is thread safe and can be called at any point after spdk_nvme_attach(). */ uint32_t spdk_nvme_ctrlr_create_ns(struct spdk_nvme_ctrlr *ctrlr, struct spdk_nvme_ns_data *payload); /** * \brief Delete a namespace. * * \param ctrlr NVMe controller to delete namespace from. * \param nsid The namespace identifier. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point after spdk_nvme_attach(). * * Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion * of commands submitted through this function. */ int spdk_nvme_ctrlr_delete_ns(struct spdk_nvme_ctrlr *ctrlr, uint32_t nsid); /** * \brief Format NVM. * * This function requests a low-level format of the media. * * \param ctrlr NVMe controller to format. * \param nsid The namespace identifier. May be SPDK_NVME_GLOBAL_NS_TAG to format all namespaces. * \param format The format information for the command. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request * * This function is thread safe and can be called at any point after spdk_nvme_attach(). */ int spdk_nvme_ctrlr_format(struct spdk_nvme_ctrlr *ctrlr, uint32_t nsid, struct spdk_nvme_format *format); /** * \brief Download a new firmware image. * * \param payload The data buffer for the firmware image. * \param size The data size will be downloaded. * \param slot The slot that the firmware image will be committed to. * * \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request, * -1 if the size is not multiple of 4. * * This function is thread safe and can be called at any point after spdk_nvme_attach(). */ int spdk_nvme_ctrlr_update_firmware(struct spdk_nvme_ctrlr *ctrlr, void *payload, uint32_t size, int slot); /** * \brief Get the identify namespace data as defined by the NVMe specification. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ const struct spdk_nvme_ns_data *spdk_nvme_ns_get_data(struct spdk_nvme_ns *ns); /** * \brief Get the namespace id (index number) from the given namespace handle. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint32_t spdk_nvme_ns_get_id(struct spdk_nvme_ns *ns); /** * \brief Determine whether a namespace is active. * * Inactive namespaces cannot be the target of I/O commands. */ bool spdk_nvme_ns_is_active(struct spdk_nvme_ns *ns); /** * \brief Get the maximum transfer size, in bytes, for an I/O sent to the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint32_t spdk_nvme_ns_get_max_io_xfer_size(struct spdk_nvme_ns *ns); /** * \brief Get the sector size, in bytes, of the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint32_t spdk_nvme_ns_get_sector_size(struct spdk_nvme_ns *ns); /** * \brief Get the number of sectors for the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint64_t spdk_nvme_ns_get_num_sectors(struct spdk_nvme_ns *ns); /** * \brief Get the size, in bytes, of the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint64_t spdk_nvme_ns_get_size(struct spdk_nvme_ns *ns); /** * \brief Get the end-to-end data protection information type of the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ enum spdk_nvme_pi_type spdk_nvme_ns_get_pi_type(struct spdk_nvme_ns *ns); /** * \brief Get the metadata size, in bytes, of the given namespace. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint32_t spdk_nvme_ns_get_md_size(struct spdk_nvme_ns *ns); /** * \brief True if the namespace can support extended LBA when end-to-end data protection enabled. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ bool spdk_nvme_ns_supports_extended_lba(struct spdk_nvme_ns *ns); /** * \brief Namespace command support flags. */ enum spdk_nvme_ns_flags { SPDK_NVME_NS_DEALLOCATE_SUPPORTED = 0x1, /**< The deallocate command is supported */ SPDK_NVME_NS_FLUSH_SUPPORTED = 0x2, /**< The flush command is supported */ SPDK_NVME_NS_RESERVATION_SUPPORTED = 0x4, /**< The reservation command is supported */ SPDK_NVME_NS_WRITE_ZEROES_SUPPORTED = 0x8, /**< The write zeroes command is supported */ SPDK_NVME_NS_DPS_PI_SUPPORTED = 0x10, /**< The end-to-end data protection is supported */ SPDK_NVME_NS_EXTENDED_LBA_SUPPORTED = 0x20, /**< The extended lba format is supported, metadata is transferred as a contiguous part of the logical block that it is associated with */ }; /** * \brief Get the flags for the given namespace. * * See spdk_nvme_ns_flags for the possible flags returned. * * This function is thread safe and can be called at any point while the controller is attached to * the SPDK NVMe driver. */ uint32_t spdk_nvme_ns_get_flags(struct spdk_nvme_ns *ns); /** * Restart the SGL walk to the specified offset when the command has scattered payloads. * * The cb_arg parameter is the value passed to readv/writev. */ typedef void (*spdk_nvme_req_reset_sgl_cb)(void *cb_arg, uint32_t offset); /** * Fill out *address and *length with the current SGL entry and advance to the next * entry for the next time the callback is invoked. * * The cb_arg parameter is the value passed to readv/writev. * The address parameter contains the virtual address of this segment. * The length parameter contains the length of this physical segment. */ typedef int (*spdk_nvme_req_next_sge_cb)(void *cb_arg, void **address, uint32_t *length); /** * \brief Submits a write I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the write I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param lba starting LBA to write the data * \param lba_count length (in sectors) for the write operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined by the SPDK_NVME_IO_FLAGS_* entries * in spdk/nvme_spec.h, for this I/O. * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_write(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags); /** * \brief Submits a write I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the write I/O * \param qpair I/O queue pair to submit the request * \param lba starting LBA to write the data * \param lba_count length (in sectors) for the write operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * \param reset_sgl_fn callback function to reset scattered payload * \param next_sge_fn callback function to iterate each scattered * payload memory segment * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_writev(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, spdk_nvme_req_reset_sgl_cb reset_sgl_fn, spdk_nvme_req_next_sge_cb next_sge_fn); /** * \brief Submits a write I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the write I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param metadata virtual address pointer to the metadata payload, the length * of metadata is specified by spdk_nvme_ns_get_md_size() * \param lba starting LBA to write the data * \param lba_count length (in sectors) for the write operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined by the SPDK_NVME_IO_FLAGS_* entries * in spdk/nvme_spec.h, for this I/O. * \param apptag_mask application tag mask. * \param apptag application tag to use end-to-end protection information. * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_write_with_md(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, void *metadata, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, uint16_t apptag_mask, uint16_t apptag); /** * \brief Submits a write zeroes I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the write zeroes I/O * \param qpair I/O queue pair to submit the request * \param lba starting LBA for this command * \param lba_count length (in sectors) for the write zero operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined by the SPDK_NVME_IO_FLAGS_* entries * in spdk/nvme_spec.h, for this I/O. * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_write_zeroes(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags); /** * \brief Submits a read I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the read I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param lba starting LBA to read the data * \param lba_count length (in sectors) for the read operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_read(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags); /** * \brief Submits a read I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the read I/O * \param qpair I/O queue pair to submit the request * \param lba starting LBA to read the data * \param lba_count length (in sectors) for the read operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * \param reset_sgl_fn callback function to reset scattered payload * \param next_sge_fn callback function to iterate each scattered * payload memory segment * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_readv(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, spdk_nvme_req_reset_sgl_cb reset_sgl_fn, spdk_nvme_req_next_sge_cb next_sge_fn); /** * \brief Submits a read I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the read I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param metadata virtual address pointer to the metadata payload, the length * of metadata is specified by spdk_nvme_ns_get_md_size() * \param lba starting LBA to read the data * \param lba_count length (in sectors) for the read operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * \param apptag_mask application tag mask. * \param apptag application tag to use end-to-end protection information. * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_read_with_md(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, void *metadata, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, uint16_t apptag_mask, uint16_t apptag); /** * \brief Submits a data set management request to the specified NVMe namespace. Data set * management operations are designed to optimize interaction with the block * translation layer inside the device. The most common type of operation is * deallocate, which is often referred to as TRIM or UNMAP. * * \param ns NVMe namespace to submit the DSM request * \param type A bit field constructed from \ref enum spdk_nvme_dsm_attribute. * \param qpair I/O queue pair to submit the request * \param ranges An array of \ref spdk_nvme_dsm_range elements describing the LBAs to operate on. * \param num_ranges The number of elements in the ranges array. * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, negated POSIX errno values otherwise. * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. * * This is a convenience wrapper that will automatically allocate and construct the correct * data buffers. Therefore, ranges does not need to be allocated from pinned memory and * can be placed on the stack. If a higher performance, zero-copy version of DSM is * required, simply build and submit a raw command using spdk_nvme_ctrlr_cmd_io_raw(). */ int spdk_nvme_ns_cmd_dataset_management(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, uint32_t type, const struct spdk_nvme_dsm_range *ranges, uint16_t num_ranges, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a flush request to the specified NVMe namespace. * * \param ns NVMe namespace to submit the flush request * \param qpair I/O queue pair to submit the request * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_flush(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a reservation register to the specified NVMe namespace. * * \param ns NVMe namespace to submit the reservation register request * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the reservation register data * \param ignore_key '1' the current reservation key check is disabled * \param action specifies the registration action * \param cptpl change the Persist Through Power Loss state * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_reservation_register(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, struct spdk_nvme_reservation_register_data *payload, bool ignore_key, enum spdk_nvme_reservation_register_action action, enum spdk_nvme_reservation_register_cptpl cptpl, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a reservation release to the specified NVMe namespace. * * \param ns NVMe namespace to submit the reservation release request * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to current reservation key * \param ignore_key '1' the current reservation key check is disabled * \param action specifies the reservation release action * \param type reservation type for the namespace * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_reservation_release(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, struct spdk_nvme_reservation_key_data *payload, bool ignore_key, enum spdk_nvme_reservation_release_action action, enum spdk_nvme_reservation_type type, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a reservation acquire to the specified NVMe namespace. * * \param ns NVMe namespace to submit the reservation acquire request * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to reservation acquire data * \param ignore_key '1' the current reservation key check is disabled * \param action specifies the reservation acquire action * \param type reservation type for the namespace * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_reservation_acquire(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, struct spdk_nvme_reservation_acquire_data *payload, bool ignore_key, enum spdk_nvme_reservation_acquire_action action, enum spdk_nvme_reservation_type type, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a reservation report to the specified NVMe namespace. * * \param ns NVMe namespace to submit the reservation report request * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer for reservation status data * \param len length bytes for reservation status data structure * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_reservation_report(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, uint32_t len, spdk_nvme_cmd_cb cb_fn, void *cb_arg); /** * \brief Submits a compare I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the compare I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param lba starting LBA to compare the data * \param lba_count length (in sectors) for the compare operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_compare(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags); /** * \brief Submits a compare I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the compare I/O * \param qpair I/O queue pair to submit the request * \param lba starting LBA to compare the data * \param lba_count length (in sectors) for the compare operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * \param reset_sgl_fn callback function to reset scattered payload * \param next_sge_fn callback function to iterate each scattered * payload memory segment * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_comparev(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, spdk_nvme_req_reset_sgl_cb reset_sgl_fn, spdk_nvme_req_next_sge_cb next_sge_fn); /** * \brief Submits a compare I/O to the specified NVMe namespace. * * \param ns NVMe namespace to submit the compare I/O * \param qpair I/O queue pair to submit the request * \param payload virtual address pointer to the data payload * \param metadata virtual address pointer to the metadata payload, the length * of metadata is specified by spdk_nvme_ns_get_md_size() * \param lba starting LBA to compare the data * \param lba_count length (in sectors) for the compare operation * \param cb_fn callback function to invoke when the I/O is completed * \param cb_arg argument to pass to the callback function * \param io_flags set flags, defined in nvme_spec.h, for this I/O * \param apptag_mask application tag mask. * \param apptag application tag to use end-to-end protection information. * * \return 0 if successfully submitted, ENOMEM if an nvme_request * structure cannot be allocated for the I/O request * * The command is submitted to a qpair allocated by spdk_nvme_ctrlr_alloc_io_qpair(). * The user must ensure that only one thread submits I/O on a given qpair at any given time. */ int spdk_nvme_ns_cmd_compare_with_md(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload, void *metadata, uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn, void *cb_arg, uint32_t io_flags, uint16_t apptag_mask, uint16_t apptag); #ifdef __cplusplus } #endif #endif