2015-09-21 15:52:41 +00:00
|
|
|
/*-
|
|
|
|
* BSD LICENSE
|
|
|
|
*
|
2016-01-26 17:47:22 +00:00
|
|
|
* Copyright (c) Intel Corporation.
|
2015-09-21 15:52:41 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2016-04-29 18:04:33 +00:00
|
|
|
/** \file
|
|
|
|
* NVMe driver public API
|
|
|
|
*/
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
#ifndef SPDK_NVME_H
|
|
|
|
#define SPDK_NVME_H
|
|
|
|
|
2016-02-12 14:52:35 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2016-02-19 19:41:12 +00:00
|
|
|
#include <stdbool.h>
|
2015-09-21 15:52:41 +00:00
|
|
|
#include <stddef.h>
|
2016-02-19 19:41:12 +00:00
|
|
|
#include <stdint.h>
|
2016-01-25 12:34:14 +00:00
|
|
|
#include "spdk/pci.h"
|
2015-09-21 15:52:41 +00:00
|
|
|
#include "nvme_spec.h"
|
|
|
|
|
2016-02-10 18:26:12 +00:00
|
|
|
#define SPDK_NVME_DEFAULT_RETRY_COUNT (4)
|
|
|
|
extern int32_t spdk_nvme_retry_count;
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-02-12 14:52:35 +00:00
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-02-10 18:26:12 +00:00
|
|
|
/** \brief Opaque handle to a controller. Returned by \ref spdk_nvme_probe()'s attach_cb. */
|
|
|
|
struct spdk_nvme_ctrlr;
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-03-07 17:36:17 +00:00
|
|
|
/**
|
|
|
|
* \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;
|
2016-05-05 01:41:16 +00:00
|
|
|
/**
|
|
|
|
* Enable submission queue in controller memory buffer
|
|
|
|
*/
|
|
|
|
bool use_cmb_sqs;
|
2016-06-14 07:19:38 +00:00
|
|
|
/**
|
|
|
|
* Type of arbitration mechanism
|
|
|
|
*/
|
|
|
|
enum spdk_nvme_cc_ams arb_mechanism;
|
2016-03-07 17:36:17 +00:00
|
|
|
};
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
2016-02-10 18:26:12 +00:00
|
|
|
* Callback for spdk_nvme_probe() enumeration.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-03-07 17:36:17 +00:00
|
|
|
* \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.
|
2016-01-29 20:15:29 +00:00
|
|
|
* \return true to attach to this device.
|
|
|
|
*/
|
2016-03-07 17:36:17 +00:00
|
|
|
typedef bool (*spdk_nvme_probe_cb)(void *cb_ctx, struct spdk_pci_device *pci_dev,
|
|
|
|
struct spdk_nvme_ctrlr_opts *opts);
|
2016-01-29 20:15:29 +00:00
|
|
|
|
|
|
|
/**
|
2016-02-10 18:26:12 +00:00
|
|
|
* Callback for spdk_nvme_probe() to report a device that has been attached to the userspace NVMe driver.
|
2016-03-07 17:36:17 +00:00
|
|
|
*
|
|
|
|
* \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.
|
2016-01-29 20:15:29 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
typedef void (*spdk_nvme_attach_cb)(void *cb_ctx, struct spdk_pci_device *pci_dev,
|
2016-03-07 17:36:17 +00:00
|
|
|
struct spdk_nvme_ctrlr *ctrlr,
|
|
|
|
const struct spdk_nvme_ctrlr_opts *opts);
|
2016-01-29 20:15:29 +00:00
|
|
|
|
2016-06-21 23:49:26 +00:00
|
|
|
/**
|
|
|
|
* Callback for spdk_nvme_probe() to report that a device attached to the userspace NVMe driver
|
|
|
|
* has been removed from the system.
|
|
|
|
*
|
|
|
|
* \param ctrlr NVMe controller instance that was removed.
|
|
|
|
*/
|
|
|
|
typedef void (*spdk_nvme_remove_cb)(void *cb_ctx, struct spdk_nvme_ctrlr *ctrlr);
|
|
|
|
|
2016-01-29 20:15:29 +00:00
|
|
|
/**
|
|
|
|
* \brief Enumerate the NVMe devices attached to the system and attach the userspace NVMe driver
|
|
|
|
* to them if desired.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param cb_ctx Opaque value which will be passed back in cb_ctx parameter of the callbacks.
|
2016-01-29 20:15:29 +00:00
|
|
|
* \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.
|
2016-06-21 23:49:26 +00:00
|
|
|
* \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.
|
2016-01-29 20:15:29 +00:00
|
|
|
*
|
|
|
|
* If called more than once, only devices that are not already attached to the SPDK NVMe driver
|
|
|
|
* will be reported.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
* To stop using the the controller and release its associated resources,
|
2016-03-15 21:08:05 +00:00
|
|
|
* call \ref spdk_nvme_detach with the spdk_nvme_ctrlr instance returned by this function.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-06-21 23:49:26 +00:00
|
|
|
int spdk_nvme_probe(void *cb_ctx,
|
|
|
|
spdk_nvme_probe_cb probe_cb,
|
|
|
|
spdk_nvme_attach_cb attach_cb,
|
|
|
|
spdk_nvme_remove_cb remove_cb);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
2016-03-15 21:08:05 +00:00
|
|
|
* \brief Detaches specified device returned by \ref spdk_nvme_probe()'s attach_cb from the NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* On success, the spdk_nvme_ctrlr handle is no longer valid.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
* This function should be called from a single thread while no other threads
|
|
|
|
* are actively using the NVMe device.
|
|
|
|
*
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_detach(struct spdk_nvme_ctrlr *ctrlr);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \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.
|
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* 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
|
2015-09-21 15:52:41 +00:00
|
|
|
* also change.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_ctrlr_reset(struct spdk_nvme_ctrlr *ctrlr);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the identify controller data as defined by the NVMe specification.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
const struct spdk_nvme_ctrlr_data *spdk_nvme_ctrlr_get_data(struct spdk_nvme_ctrlr *ctrlr);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-07-06 21:55:28 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \brief Get the number of namespaces for the given NVMe controller.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* This is equivalent to calling spdk_nvme_ctrlr_get_data() to get the
|
|
|
|
* spdk_nvme_ctrlr_data and then reading the nn field.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t spdk_nvme_ctrlr_get_num_ns(struct spdk_nvme_ctrlr *ctrlr);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-01-06 05:43:33 +00:00
|
|
|
/**
|
2016-01-15 19:54:53 +00:00
|
|
|
* \brief Determine if a particular log page is supported by the given NVMe controller.
|
2016-01-06 05:43:33 +00:00
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2016-01-15 19:54:53 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* \sa spdk_nvme_ctrlr_cmd_get_log_page()
|
2016-01-06 05:43:33 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
bool spdk_nvme_ctrlr_is_log_page_supported(struct spdk_nvme_ctrlr *ctrlr, uint8_t log_page);
|
2016-01-06 05:43:33 +00:00
|
|
|
|
2016-01-25 05:04:23 +00:00
|
|
|
/**
|
|
|
|
* \brief Determine if a particular feature is supported by the given NVMe controller.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2016-01-25 05:04:23 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* \sa spdk_nvme_ctrlr_cmd_get_feature()
|
2016-01-25 05:04:23 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
bool spdk_nvme_ctrlr_is_feature_supported(struct spdk_nvme_ctrlr *ctrlr, uint8_t feature_code);
|
2016-01-25 05:04:23 +00:00
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* Signature for callback function invoked when a command is completed.
|
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* The spdk_nvme_cpl parameter contains the completion status.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
typedef void (*spdk_nvme_cmd_cb)(void *, const struct spdk_nvme_cpl *);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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
|
2016-02-10 18:26:12 +00:00
|
|
|
* spdk_nvme_register_aer_callback().
|
|
|
|
* The spdk_nvme_cpl parameter contains the completion status of the
|
2015-09-21 15:52:41 +00:00
|
|
|
* asynchronous event request that was completed.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
typedef void (*spdk_nvme_aer_cb)(void *aer_cb_arg,
|
2016-02-09 18:06:48 +00:00
|
|
|
const struct spdk_nvme_cpl *);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-02-10 18:26:12 +00:00
|
|
|
void spdk_nvme_ctrlr_register_aer_callback(struct spdk_nvme_ctrlr *ctrlr,
|
|
|
|
spdk_nvme_aer_cb aer_cb_fn,
|
|
|
|
void *aer_cb_arg);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-02-29 21:19:02 +00:00
|
|
|
/**
|
|
|
|
* \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;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \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
|
2016-02-10 18:26:12 +00:00
|
|
|
* the spdk_nvme_ns_cmd_* functions instead. The validity of the command will
|
2015-09-21 15:52:41 +00:00
|
|
|
* 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.
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_ctrlr_cmd_io_raw(struct spdk_nvme_ctrlr *ctrlr,
|
2016-02-29 21:19:02 +00:00
|
|
|
struct spdk_nvme_qpair *qpair,
|
2016-02-10 18:26:12 +00:00
|
|
|
struct spdk_nvme_cmd *cmd,
|
|
|
|
void *buf, uint32_t len,
|
|
|
|
spdk_nvme_cmd_cb cb_fn, void *cb_arg);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
2016-02-29 21:19:02 +00:00
|
|
|
* \brief Process any outstanding completions for I/O submitted on a queue pair.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* This call is non-blocking, i.e. it only
|
2015-09-21 15:52:41 +00:00
|
|
|
* processes completions that are ready at the time of this function call. It does not
|
2015-11-05 22:00:20 +00:00
|
|
|
* wait for outstanding commands to finish.
|
|
|
|
*
|
2016-03-29 17:26:28 +00:00
|
|
|
* For each completed command, the request's callback function will
|
|
|
|
* be called if specified as non-NULL when the request was submitted.
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair Queue pair to check for completions.
|
2015-11-05 22:00:20 +00:00
|
|
|
* \param max_completions Limit the number of completions to be processed in one call, or 0
|
|
|
|
* for unlimited.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-01-11 16:10:02 +00:00
|
|
|
* \return Number of completions processed (may be 0) or negative on error.
|
|
|
|
*
|
2016-03-29 17:26:28 +00:00
|
|
|
* \sa spdk_nvme_cmd_cb
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* This function may be called at any point while the controller is attached to
|
2016-01-29 20:15:29 +00:00
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* The caller must ensure that each queue pair is only used from one thread at a time.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
int32_t spdk_nvme_qpair_process_completions(struct spdk_nvme_qpair *qpair,
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t max_completions);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Send the given admin command to the NVMe controller.
|
|
|
|
*
|
|
|
|
* This is a low level interface for submitting admin commands directly. Prefer
|
2016-02-10 18:26:12 +00:00
|
|
|
* the spdk_nvme_ctrlr_cmd_* functions instead. The validity of the command will
|
2015-09-21 15:52:41 +00:00
|
|
|
* 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.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion
|
2015-09-21 15:52:41 +00:00
|
|
|
* of commands submitted through this function.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \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.
|
|
|
|
*
|
2016-01-11 16:10:02 +00:00
|
|
|
* \return Number of completions processed (may be 0) or negative on error.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int32_t spdk_nvme_ctrlr_process_admin_completions(struct spdk_nvme_ctrlr *ctrlr);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
|
2016-02-10 18:26:12 +00:00
|
|
|
/** \brief Opaque handle to a namespace. Obtained by calling spdk_nvme_ctrlr_get_ns(). */
|
|
|
|
struct spdk_nvme_ns;
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \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
|
2016-02-10 18:26:12 +00:00
|
|
|
* spdk_nvme_ctrlr_get_num_ns().
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
struct spdk_nvme_ns *spdk_nvme_ctrlr_get_ns(struct spdk_nvme_ctrlr *ctrlr, uint32_t ns_id);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-01-06 05:43:33 +00:00
|
|
|
/**
|
2016-01-15 19:54:53 +00:00
|
|
|
* \brief Get a specific log page from the NVMe controller.
|
2016-01-06 05:43:33 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to query.
|
2016-01-15 19:54:53 +00:00
|
|
|
* \param log_page The log page identifier.
|
2016-02-10 18:26:12 +00:00
|
|
|
* \param nsid Depending on the log page, this may be 0, a namespace identifier, or SPDK_NVME_GLOBAL_NS_TAG.
|
2016-01-15 19:54:53 +00:00
|
|
|
* \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 log page has been retrieved.
|
|
|
|
* \param cb_arg Argument to pass to the callback function.
|
2016-01-15 19:49:16 +00:00
|
|
|
*
|
|
|
|
* \return 0 if successfully submitted, ENOMEM if resources could not be allocated for this request
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2016-01-15 19:54:53 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion
|
2016-01-06 05:43:33 +00:00
|
|
|
* of commands submitted through this function.
|
2016-01-15 19:54:53 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* \sa spdk_nvme_ctrlr_is_log_page_supported()
|
2016-01-06 05:43:33 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
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,
|
|
|
|
spdk_nvme_cmd_cb cb_fn, void *cb_arg);
|
2016-01-06 05:43:33 +00:00
|
|
|
|
2016-01-16 04:04:52 +00:00
|
|
|
/**
|
|
|
|
* \brief Set specific feature for the given NVMe controller.
|
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to manipulate.
|
2016-01-16 04:04:52 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2016-01-16 04:04:52 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion
|
2016-01-16 04:04:52 +00:00
|
|
|
* of commands submitted through this function.
|
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* \sa spdk_nvme_ctrlr_cmd_get_feature()
|
2016-01-16 04:04:52 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2016-01-16 04:04:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get specific feature from given NVMe controller.
|
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to query.
|
2016-01-16 04:04:52 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2016-01-16 04:04:52 +00:00
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* Call \ref spdk_nvme_ctrlr_process_admin_completions() to poll for completion
|
2016-01-16 04:04:52 +00:00
|
|
|
* of commands submitted through this function.
|
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* \sa spdk_nvme_ctrlr_cmd_set_feature()
|
2016-01-16 04:04:52 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2016-01-16 04:04:52 +00:00
|
|
|
|
2016-02-25 03:44:44 +00:00
|
|
|
/**
|
2016-03-29 18:19:02 +00:00
|
|
|
* \brief Attach the specified namespace to controllers.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to use for command submission.
|
|
|
|
* \param nsid Namespace identifier for namespace to attach.
|
2016-02-25 03:44:44 +00:00
|
|
|
* \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);
|
|
|
|
|
|
|
|
/**
|
2016-03-29 18:19:02 +00:00
|
|
|
* \brief Detach the specified namespace from controllers.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to use for command submission.
|
|
|
|
* \param nsid Namespace ID to detach.
|
2016-02-25 03:44:44 +00:00
|
|
|
* \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);
|
|
|
|
|
|
|
|
/**
|
2016-03-29 18:19:02 +00:00
|
|
|
* \brief Create a namespace.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to create namespace on.
|
|
|
|
* \param payload The pointer to the NVMe namespace data.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
2016-05-19 00:05:38 +00:00
|
|
|
* \return Namespace ID (>= 1) if successfully created, or 0 if the request failed.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
|
|
|
* This function is thread safe and can be called at any point after spdk_nvme_attach().
|
|
|
|
*/
|
2016-05-19 00:05:38 +00:00
|
|
|
uint32_t spdk_nvme_ctrlr_create_ns(struct spdk_nvme_ctrlr *ctrlr,
|
|
|
|
struct spdk_nvme_ns_data *payload);
|
2016-02-25 03:44:44 +00:00
|
|
|
|
|
|
|
/**
|
2016-03-29 18:19:02 +00:00
|
|
|
* \brief Delete a namespace.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* \param ctrlr NVMe controller to delete namespace from.
|
|
|
|
* \param nsid The namespace identifier.
|
2016-02-25 03:44:44 +00:00
|
|
|
*
|
|
|
|
* \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);
|
|
|
|
|
2016-03-07 06:29:50 +00:00
|
|
|
/**
|
2016-03-29 18:19:02 +00:00
|
|
|
* \brief Format NVM.
|
2016-03-07 06:29:50 +00:00
|
|
|
*
|
2016-03-29 18:19:02 +00:00
|
|
|
* 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.
|
2016-03-07 06:29:50 +00:00
|
|
|
*
|
|
|
|
* \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);
|
|
|
|
|
2016-05-03 05:18:39 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \brief Get the identify namespace data as defined by the NVMe specification.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
const struct spdk_nvme_ns_data *spdk_nvme_ns_get_data(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the namespace id (index number) from the given namespace handle.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t spdk_nvme_ns_get_id(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-03-01 21:39:24 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \brief Get the maximum transfer size, in bytes, for an I/O sent to the given namespace.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t spdk_nvme_ns_get_max_io_xfer_size(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the sector size, in bytes, of the given namespace.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t spdk_nvme_ns_get_sector_size(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the number of sectors for the given namespace.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint64_t spdk_nvme_ns_get_num_sectors(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the size, in bytes, of the given namespace.
|
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint64_t spdk_nvme_ns_get_size(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-04-07 06:52:43 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-12-30 22:52:17 +00:00
|
|
|
/**
|
|
|
|
* \brief Namespace command support flags.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
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 */
|
2016-04-07 06:52:43 +00:00
|
|
|
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 */
|
2015-09-21 15:52:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Get the flags for the given namespace.
|
|
|
|
*
|
2016-02-10 18:26:12 +00:00
|
|
|
* See spdk_nvme_ns_flags for the possible flags returned.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-01-29 20:15:29 +00:00
|
|
|
* This function is thread safe and can be called at any point while the controller is attached to
|
|
|
|
* the SPDK NVMe driver.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t spdk_nvme_ns_get_flags(struct spdk_nvme_ns *ns);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2015-12-04 03:52:37 +00:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
typedef void (*spdk_nvme_req_reset_sgl_cb)(void *cb_arg, uint32_t offset);
|
2015-12-04 03:52:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 physical address of this segment.
|
|
|
|
* The length parameter contains the length of this physical segment.
|
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
typedef int (*spdk_nvme_req_next_sge_cb)(void *cb_arg, uint64_t *address, uint32_t *length);
|
2015-12-04 03:52:37 +00:00
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a write I/O to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the write I/O
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2015-09-21 15:52:41 +00:00
|
|
|
* \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
|
2016-02-10 18:26:12 +00:00
|
|
|
* \param io_flags set flags, defined by the SPDK_NVME_IO_FLAGS_* entries
|
2015-12-30 22:52:17 +00:00
|
|
|
* in spdk/nvme_spec.h, for this I/O.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
* \return 0 if successfully submitted, ENOMEM if an nvme_request
|
|
|
|
* structure cannot be allocated for the I/O request
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
int spdk_nvme_ns_cmd_write(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload,
|
2016-02-10 18:26:12 +00:00
|
|
|
uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn,
|
|
|
|
void *cb_arg, uint32_t io_flags);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2015-12-04 03:52:37 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a write I/O to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the write I/O
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2015-12-04 03:52:37 +00:00
|
|
|
* \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
|
2015-12-30 22:52:17 +00:00
|
|
|
* \param io_flags set flags, defined in nvme_spec.h, for this I/O
|
2015-12-04 03:52:37 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-12-04 03:52:37 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
int spdk_nvme_ns_cmd_writev(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair,
|
|
|
|
uint64_t lba, uint32_t lba_count,
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2015-12-04 03:52:37 +00:00
|
|
|
|
2016-04-07 06:52:43 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2016-01-12 07:43:24 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a write zeroes I/O to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the write zeroes I/O
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-01-12 07:43:24 +00:00
|
|
|
* \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
|
2016-02-10 18:26:12 +00:00
|
|
|
* \param io_flags set flags, defined by the SPDK_NVME_IO_FLAGS_* entries
|
2016-01-12 07:43:24 +00:00
|
|
|
* 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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-01-12 07:43:24 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
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,
|
2016-02-10 18:26:12 +00:00
|
|
|
uint32_t io_flags);
|
2016-01-12 07:43:24 +00:00
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a read I/O to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the read I/O
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2015-09-21 15:52:41 +00:00
|
|
|
* \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
|
2015-12-30 22:52:17 +00:00
|
|
|
* \param io_flags set flags, defined in nvme_spec.h, for this I/O
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
|
|
|
* \return 0 if successfully submitted, ENOMEM if an nvme_request
|
|
|
|
* structure cannot be allocated for the I/O request
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
int spdk_nvme_ns_cmd_read(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair, void *payload,
|
2016-02-10 18:26:12 +00:00
|
|
|
uint64_t lba, uint32_t lba_count, spdk_nvme_cmd_cb cb_fn,
|
|
|
|
void *cb_arg, uint32_t io_flags);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2015-12-04 03:52:37 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a read I/O to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the read I/O
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2015-12-04 03:52:37 +00:00
|
|
|
* \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
|
2015-12-30 22:52:17 +00:00
|
|
|
* \param io_flags set flags, defined in nvme_spec.h, for this I/O
|
2015-12-04 03:52:37 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-12-04 03:52:37 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
int spdk_nvme_ns_cmd_readv(struct spdk_nvme_ns *ns, struct spdk_nvme_qpair *qpair,
|
|
|
|
uint64_t lba, uint32_t lba_count,
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2015-12-04 03:52:37 +00:00
|
|
|
|
2016-04-07 06:52:43 +00:00
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
/**
|
2016-10-03 17:37:48 +00:00
|
|
|
* \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.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-10-03 17:37:48 +00:00
|
|
|
* \param ns NVMe namespace to submit the DSM request
|
|
|
|
* \param type A bit field constructed from \ref enum spdk_nvme_dsm_attribute.
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-10-03 17:37:48 +00:00
|
|
|
* \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.
|
2015-09-21 15:52:41 +00:00
|
|
|
* \param cb_fn callback function to invoke when the I/O is completed
|
|
|
|
* \param cb_arg argument to pass to the callback function
|
|
|
|
*
|
2016-10-03 17:37:48 +00:00
|
|
|
* \return 0 if successfully submitted, negated POSIX errno values otherwise.
|
2015-09-21 15:52:41 +00:00
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-10-03 17:37:48 +00:00
|
|
|
*
|
|
|
|
* 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);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Submits a flush request to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the flush request
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2015-09-21 15:52:41 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2015-09-21 15:52:41 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
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);
|
2015-09-21 15:52:41 +00:00
|
|
|
|
2016-01-22 08:39:55 +00:00
|
|
|
/**
|
|
|
|
* \brief Submits a reservation register to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the reservation register request
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-01-22 08:39:55 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-01-22 08:39:55 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_ns_cmd_reservation_register(struct spdk_nvme_ns *ns,
|
2016-02-29 21:19:02 +00:00
|
|
|
struct spdk_nvme_qpair *qpair,
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2016-01-22 08:39:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Submits a reservation release to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the reservation release request
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-01-22 08:39:55 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-01-22 08:39:55 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_ns_cmd_reservation_release(struct spdk_nvme_ns *ns,
|
2016-02-29 21:19:02 +00:00
|
|
|
struct spdk_nvme_qpair *qpair,
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2016-01-22 08:39:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Submits a reservation acquire to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the reservation acquire request
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-01-22 08:39:55 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-01-22 08:39:55 +00:00
|
|
|
*/
|
2016-02-10 18:26:12 +00:00
|
|
|
int spdk_nvme_ns_cmd_reservation_acquire(struct spdk_nvme_ns *ns,
|
2016-02-29 21:19:02 +00:00
|
|
|
struct spdk_nvme_qpair *qpair,
|
2016-02-10 18:26:12 +00:00
|
|
|
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);
|
2016-01-22 08:39:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Submits a reservation report to the specified NVMe namespace.
|
|
|
|
*
|
|
|
|
* \param ns NVMe namespace to submit the reservation report request
|
2016-02-29 21:19:02 +00:00
|
|
|
* \param qpair I/O queue pair to submit the request
|
2016-01-22 08:39:55 +00:00
|
|
|
* \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
|
|
|
|
*
|
2016-02-29 21:19:02 +00:00
|
|
|
* 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.
|
2016-01-22 08:39:55 +00:00
|
|
|
*/
|
2016-02-29 21:19:02 +00:00
|
|
|
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);
|
2016-01-22 08:39:55 +00:00
|
|
|
|
2015-09-21 15:52:41 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|