This patch also drops support for automatically unbinding devices from the kernel - run scripts/setup.sh first. Our generic pci interface is now hidden behind include/spdk/pci.h and implemented in lib/util/pci.c. We no longer wrap the calls in nvme_impl.h or ioat_impl.h. The implementation now only uses DPDK and the libpciaccess dependency has been removed. If using a version of DPDK earlier than 16.07, enumerating devices by class code isn't available and only Intel SSDs will be discovered. DPDK 16.07 adds enumeration by class code and all NVMe devices will be correctly discovered. Change-Id: I0e8bac36b5ca57df604a2b310c47342c67dc9f3c Signed-off-by: Ben Walker <benjamin.walker@intel.com>
212 lines
5.8 KiB
C
212 lines
5.8 KiB
C
/*-
|
|
* 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 integration callbacks
|
|
*
|
|
* This file describes the callback functions required to integrate
|
|
* the userspace NVMe driver for a specific implementation. This
|
|
* implementation is specific for DPDK. Users would
|
|
* revise it as necessary for their own particular environment if not
|
|
* using it within the DPDK framework.
|
|
*/
|
|
|
|
#ifndef __NVME_IMPL_H__
|
|
#define __NVME_IMPL_H__
|
|
|
|
#include "spdk/vtophys.h"
|
|
#include "spdk/pci.h"
|
|
#include "spdk/nvme_spec.h"
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdbool.h>
|
|
#include <rte_config.h>
|
|
#include <rte_cycles.h>
|
|
#include <rte_malloc.h>
|
|
#include <rte_mempool.h>
|
|
#include <rte_version.h>
|
|
#include <rte_memzone.h>
|
|
#include <rte_eal.h>
|
|
|
|
#include <rte_pci.h>
|
|
|
|
#include "spdk/pci_ids.h"
|
|
|
|
/**
|
|
* \page nvme_driver_integration NVMe Driver Integration
|
|
*
|
|
* Users can integrate the userspace NVMe driver into their environment
|
|
* by implementing the callbacks in nvme_impl.h. These callbacks
|
|
* enable users to specify how to allocate pinned and physically
|
|
* contiguous memory, performance virtual to physical address
|
|
* translations, log messages, PCI configuration and register mapping,
|
|
* and a number of other facilities that may differ depending on the
|
|
* environment.
|
|
*/
|
|
|
|
/**
|
|
* Allocate a pinned, physically contiguous memory buffer with the
|
|
* given size and alignment.
|
|
* Note: these calls are only made during driver initialization.
|
|
*/
|
|
static inline void *
|
|
nvme_malloc(const char *tag, size_t size, unsigned align, uint64_t *phys_addr)
|
|
{
|
|
void *buf = rte_malloc(tag, size, align);
|
|
if (buf) {
|
|
memset(buf, 0, size);
|
|
*phys_addr = rte_malloc_virt2phy(buf);
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
/**
|
|
* Free a memory buffer previously allocated with nvme_malloc.
|
|
*/
|
|
#define nvme_free(buf) rte_free(buf)
|
|
|
|
/**
|
|
* Reserve a named, process shared memory zone with the given size,
|
|
* socket_id and flags.
|
|
* Return a pointer to the allocated memory address. If the allocation
|
|
* cannot be done, return NULL.
|
|
*/
|
|
static inline void *
|
|
nvme_memzone_reserve(const char *name, size_t len, int socket_id, unsigned flags)
|
|
{
|
|
const struct rte_memzone *mz = rte_memzone_reserve(name, len, socket_id, flags);
|
|
|
|
if (mz != NULL) {
|
|
return mz->addr;
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Lookup the memory zone identified by the given name.
|
|
* Return a pointer to the reserved memory address. If the reservation
|
|
* cannot be found, return NULL.
|
|
*/
|
|
static inline void *
|
|
nvme_memzone_lookup(const char *name)
|
|
{
|
|
const struct rte_memzone *mz = rte_memzone_lookup(name);
|
|
|
|
if (mz != NULL) {
|
|
return mz->addr;
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Free the memory zone identified by the given name.
|
|
*/
|
|
static inline int
|
|
nvme_memzone_free(const char *name)
|
|
{
|
|
const struct rte_memzone *mz = rte_memzone_lookup(name);
|
|
|
|
if (mz != NULL) {
|
|
return rte_memzone_free(mz);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Return true if the calling process is primary process
|
|
*/
|
|
static inline bool
|
|
nvme_process_is_primary(void)
|
|
{
|
|
return (rte_eal_process_type() == RTE_PROC_PRIMARY);
|
|
}
|
|
|
|
/**
|
|
* Return the physical address for the specified virtual address.
|
|
*/
|
|
#define nvme_vtophys(buf) spdk_vtophys(buf)
|
|
#define NVME_VTOPHYS_ERROR SPDK_VTOPHYS_ERROR
|
|
|
|
typedef struct rte_mempool nvme_mempool_t;
|
|
|
|
/**
|
|
* Create a mempool with the given configuration.
|
|
* Return a pointer to the allocated memory address. If the allocation
|
|
* cannot be done, return NULL.
|
|
*/
|
|
static inline nvme_mempool_t *
|
|
nvme_mempool_create(const char *name, unsigned n, unsigned elt_size,
|
|
unsigned cache_size)
|
|
{
|
|
struct rte_mempool *mp;
|
|
|
|
mp = rte_mempool_create(name, n, elt_size, cache_size,
|
|
0, NULL, NULL, NULL, NULL,
|
|
SOCKET_ID_ANY, 0);
|
|
|
|
if (mp == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
return (nvme_mempool_t *)mp;
|
|
}
|
|
|
|
static inline void
|
|
nvme_mempool_get(nvme_mempool_t *mp, void **buf)
|
|
{
|
|
rte_mempool_get(mp, buf);
|
|
}
|
|
|
|
static inline void
|
|
nvme_mempool_put(nvme_mempool_t *mp, void *buf)
|
|
{
|
|
rte_mempool_put(mp, buf);
|
|
}
|
|
|
|
/**
|
|
* Get a monotonic timestamp counter (used for measuring timeouts during initialization).
|
|
*/
|
|
#define nvme_get_tsc() rte_get_timer_cycles()
|
|
|
|
/**
|
|
* Get the tick rate of nvme_get_tsc() per second.
|
|
*/
|
|
#define nvme_get_tsc_hz() rte_get_timer_hz()
|
|
|
|
#endif /* __NVME_IMPL_H__ */
|