2018-01-29 14:11:31 +01:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
* Copyright 2017 6WIND S.A.
|
2018-03-20 21:20:35 +02:00
|
|
|
* Copyright 2017 Mellanox Technologies, Ltd
|
2017-09-01 10:06:49 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MLX4_UTILS_H_
|
|
|
|
#define MLX4_UTILS_H_
|
|
|
|
|
2017-11-02 14:41:25 +01:00
|
|
|
#include <assert.h>
|
2017-09-01 10:06:58 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2017-09-01 10:06:49 +02:00
|
|
|
#include <rte_common.h>
|
|
|
|
#include <rte_log.h>
|
|
|
|
|
|
|
|
#include "mlx4.h"
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When debugging is enabled (NDEBUG not defined), file, line and function
|
|
|
|
* information replace the driver name (MLX4_DRIVER_NAME) in log messages.
|
|
|
|
*/
|
|
|
|
|
2017-10-12 14:19:18 +02:00
|
|
|
/** Return the file name part of a path. */
|
2017-09-01 10:06:49 +02:00
|
|
|
static inline const char *
|
|
|
|
pmd_drv_log_basename(const char *s)
|
|
|
|
{
|
|
|
|
const char *n = s;
|
|
|
|
|
|
|
|
while (*n)
|
|
|
|
if (*(n++) == '/')
|
|
|
|
s = n;
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PMD_DRV_LOG(level, ...) \
|
|
|
|
RTE_LOG(level, PMD, \
|
|
|
|
RTE_FMT("%s:%u: %s(): " RTE_FMT_HEAD(__VA_ARGS__,) "\n", \
|
|
|
|
pmd_drv_log_basename(__FILE__), \
|
|
|
|
__LINE__, \
|
|
|
|
__func__, \
|
|
|
|
RTE_FMT_TAIL(__VA_ARGS__,)))
|
|
|
|
#define DEBUG(...) PMD_DRV_LOG(DEBUG, __VA_ARGS__)
|
|
|
|
#define claim_zero(...) assert((__VA_ARGS__) == 0)
|
|
|
|
|
|
|
|
#else /* NDEBUG */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Like assert(), DEBUG() becomes a no-op and claim_zero() does not perform
|
|
|
|
* any check when debugging is disabled.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define PMD_DRV_LOG(level, ...) \
|
|
|
|
RTE_LOG(level, PMD, \
|
|
|
|
RTE_FMT(MLX4_DRIVER_NAME ": " \
|
|
|
|
RTE_FMT_HEAD(__VA_ARGS__,) "\n", \
|
|
|
|
RTE_FMT_TAIL(__VA_ARGS__,)))
|
|
|
|
#define DEBUG(...) (void)0
|
|
|
|
#define claim_zero(...) (__VA_ARGS__)
|
|
|
|
|
|
|
|
#endif /* NDEBUG */
|
|
|
|
|
|
|
|
#define INFO(...) PMD_DRV_LOG(INFO, __VA_ARGS__)
|
|
|
|
#define WARN(...) PMD_DRV_LOG(WARNING, __VA_ARGS__)
|
|
|
|
#define ERROR(...) PMD_DRV_LOG(ERR, __VA_ARGS__)
|
|
|
|
|
2017-10-12 14:19:18 +02:00
|
|
|
/** Allocate a buffer on the stack and fill it with a printf format string. */
|
2017-09-01 10:06:58 +02:00
|
|
|
#define MKSTR(name, ...) \
|
|
|
|
char name[snprintf(NULL, 0, __VA_ARGS__) + 1]; \
|
|
|
|
\
|
|
|
|
snprintf(name, sizeof(name), __VA_ARGS__)
|
|
|
|
|
2017-10-12 14:19:19 +02:00
|
|
|
/** Generate a string out of the provided arguments. */
|
|
|
|
#define MLX4_STR(...) # __VA_ARGS__
|
|
|
|
|
|
|
|
/** Similar to MLX4_STR() with enclosed macros expanded first. */
|
|
|
|
#define MLX4_STR_EXPAND(...) MLX4_STR(__VA_ARGS__)
|
|
|
|
|
net/mlx4: add iovec-like allocation wrappers
These wrappers implement the ability to allocate room for several disparate
objects as a single contiguous allocation while complying with their
respective alignment constraints.
This is usually more efficient than allocating and freeing them
individually if they are not expected to be reallocated with rte_realloc().
A typical use case is when several objects that cannot be dissociated must
be allocated together, as shown in the following example:
struct b {
...
struct d *d;
}
struct a {
...
struct b *b;
struct c *c;
}
struct mlx4_malloc_vec vec[] = {
{ .size = sizeof(struct a), .addr = &ptr_a, },
{ .size = sizeof(struct b), .addr = &ptr_b, },
{ .size = sizeof(struct c), .addr = &ptr_c, },
{ .size = sizeof(struct d), .addr = &ptr_d, },
};
if (!mlx4_mallocv(NULL, vec, RTE_DIM(vec)))
goto error;
struct a *a = ptr_a;
a->b = ptr_b;
a->c = ptr_c;
a->b->d = ptr_d;
...
rte_free(a);
Signed-off-by: Adrien Mazarguil <adrien.mazarguil@6wind.com>
Acked-by: Nelio Laranjeiro <nelio.laranjeiro@6wind.com>
2017-10-12 14:19:23 +02:00
|
|
|
/** Object description used with mlx4_mallocv() and similar functions. */
|
|
|
|
struct mlx4_malloc_vec {
|
|
|
|
size_t align; /**< Alignment constraint (power of 2), 0 if unknown. */
|
|
|
|
size_t size; /**< Object size. */
|
|
|
|
void **addr; /**< Storage for allocation address. */
|
|
|
|
};
|
|
|
|
|
2017-09-01 10:06:52 +02:00
|
|
|
/* mlx4_utils.c */
|
|
|
|
|
|
|
|
int mlx4_fd_set_non_blocking(int fd);
|
net/mlx4: add iovec-like allocation wrappers
These wrappers implement the ability to allocate room for several disparate
objects as a single contiguous allocation while complying with their
respective alignment constraints.
This is usually more efficient than allocating and freeing them
individually if they are not expected to be reallocated with rte_realloc().
A typical use case is when several objects that cannot be dissociated must
be allocated together, as shown in the following example:
struct b {
...
struct d *d;
}
struct a {
...
struct b *b;
struct c *c;
}
struct mlx4_malloc_vec vec[] = {
{ .size = sizeof(struct a), .addr = &ptr_a, },
{ .size = sizeof(struct b), .addr = &ptr_b, },
{ .size = sizeof(struct c), .addr = &ptr_c, },
{ .size = sizeof(struct d), .addr = &ptr_d, },
};
if (!mlx4_mallocv(NULL, vec, RTE_DIM(vec)))
goto error;
struct a *a = ptr_a;
a->b = ptr_b;
a->c = ptr_c;
a->b->d = ptr_d;
...
rte_free(a);
Signed-off-by: Adrien Mazarguil <adrien.mazarguil@6wind.com>
Acked-by: Nelio Laranjeiro <nelio.laranjeiro@6wind.com>
2017-10-12 14:19:23 +02:00
|
|
|
size_t mlx4_mallocv(const char *type, const struct mlx4_malloc_vec *vec,
|
|
|
|
unsigned int cnt);
|
|
|
|
size_t mlx4_zmallocv(const char *type, const struct mlx4_malloc_vec *vec,
|
|
|
|
unsigned int cnt);
|
|
|
|
size_t mlx4_mallocv_socket(const char *type, const struct mlx4_malloc_vec *vec,
|
|
|
|
unsigned int cnt, int socket);
|
|
|
|
size_t mlx4_zmallocv_socket(const char *type, const struct mlx4_malloc_vec *vec,
|
|
|
|
unsigned int cnt, int socket);
|
2017-09-01 10:06:52 +02:00
|
|
|
|
2017-09-01 10:06:49 +02:00
|
|
|
#endif /* MLX4_UTILS_H_ */
|