b8b1f19222
Now that multiple trace entries can be chained together to form a larger argument buffer, we can reduce the size of a single entry back to 32 bytes, while still allowing the user to pass multiple parameters. Signed-off-by: Konrad Sztyber <konrad.sztyber@intel.com> Change-Id: Ic92a1413498df28a8561a13c0f06d895d0af2cc6 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/8407 Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Community-CI: Mellanox Build Bot Reviewed-by: Tomasz Zawadzki <tomasz.zawadzki@intel.com> Reviewed-by: Jim Harris <james.r.harris@intel.com>
435 lines
13 KiB
C
435 lines
13 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
|
|
* Tracepoint library
|
|
*/
|
|
|
|
#ifndef _SPDK_TRACE_H_
|
|
#define _SPDK_TRACE_H_
|
|
|
|
#include "spdk/stdinc.h"
|
|
#include "spdk/assert.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#define SPDK_DEFAULT_NUM_TRACE_ENTRIES (32 * 1024)
|
|
|
|
struct spdk_trace_entry {
|
|
uint64_t tsc;
|
|
uint16_t tpoint_id;
|
|
uint16_t poller_id;
|
|
uint32_t size;
|
|
uint64_t object_id;
|
|
uint8_t args[8];
|
|
};
|
|
|
|
struct spdk_trace_entry_buffer {
|
|
uint64_t tsc;
|
|
uint16_t tpoint_id;
|
|
uint8_t data[22];
|
|
};
|
|
|
|
SPDK_STATIC_ASSERT(sizeof(struct spdk_trace_entry_buffer) == sizeof(struct spdk_trace_entry),
|
|
"Invalid size of trace entry buffer");
|
|
|
|
/* If type changes from a uint8_t, change this value. */
|
|
#define SPDK_TRACE_MAX_OWNER (UCHAR_MAX + 1)
|
|
|
|
struct spdk_trace_owner {
|
|
uint8_t type;
|
|
char id_prefix;
|
|
};
|
|
|
|
/* If type changes from a uint8_t, change this value. */
|
|
#define SPDK_TRACE_MAX_OBJECT (UCHAR_MAX + 1)
|
|
|
|
struct spdk_trace_object {
|
|
uint8_t type;
|
|
char id_prefix;
|
|
};
|
|
|
|
#define SPDK_TRACE_MAX_GROUP_ID 16
|
|
#define SPDK_TRACE_MAX_TPOINT_ID (SPDK_TRACE_MAX_GROUP_ID * 64)
|
|
#define SPDK_TPOINT_ID(group, tpoint) ((group * 64) + tpoint)
|
|
|
|
#define SPDK_TRACE_ARG_TYPE_INT 0
|
|
#define SPDK_TRACE_ARG_TYPE_PTR 1
|
|
#define SPDK_TRACE_ARG_TYPE_STR 2
|
|
|
|
#define SPDK_TRACE_MAX_ARGS_COUNT 5
|
|
|
|
struct spdk_trace_argument {
|
|
char name[14];
|
|
uint8_t type;
|
|
uint8_t size;
|
|
};
|
|
|
|
struct spdk_trace_tpoint {
|
|
char name[24];
|
|
uint16_t tpoint_id;
|
|
uint8_t owner_type;
|
|
uint8_t object_type;
|
|
uint8_t new_object;
|
|
uint8_t num_args;
|
|
struct spdk_trace_argument args[SPDK_TRACE_MAX_ARGS_COUNT];
|
|
};
|
|
|
|
struct spdk_trace_history {
|
|
/** Logical core number associated with this structure instance. */
|
|
int lcore;
|
|
|
|
/** Number of trace_entries contained in each trace_history. */
|
|
uint64_t num_entries;
|
|
|
|
/**
|
|
* Running count of number of occurrences of each tracepoint on this
|
|
* lcore. Debug tools can use this to easily count tracepoints such as
|
|
* number of SCSI tasks completed or PDUs read.
|
|
*/
|
|
uint64_t tpoint_count[SPDK_TRACE_MAX_TPOINT_ID];
|
|
|
|
/** Index to next spdk_trace_entry to fill. */
|
|
uint64_t next_entry;
|
|
|
|
/**
|
|
* Circular buffer of spdk_trace_entry structures for tracing
|
|
* tpoints on this core. Debug tool spdk_trace reads this
|
|
* buffer from shared memory to post-process the tpoint entries and
|
|
* display in a human-readable format.
|
|
*/
|
|
struct spdk_trace_entry entries[0];
|
|
};
|
|
|
|
#define SPDK_TRACE_MAX_LCORE 128
|
|
|
|
struct spdk_trace_flags {
|
|
uint64_t tsc_rate;
|
|
uint64_t tpoint_mask[SPDK_TRACE_MAX_GROUP_ID];
|
|
struct spdk_trace_owner owner[UCHAR_MAX + 1];
|
|
struct spdk_trace_object object[UCHAR_MAX + 1];
|
|
struct spdk_trace_tpoint tpoint[SPDK_TRACE_MAX_TPOINT_ID];
|
|
|
|
/** Offset of each trace_history from the beginning of this data structure.
|
|
* The last one is the offset of the file end.
|
|
*/
|
|
uint64_t lcore_history_offsets[SPDK_TRACE_MAX_LCORE + 1];
|
|
};
|
|
extern struct spdk_trace_flags *g_trace_flags;
|
|
extern struct spdk_trace_histories *g_trace_histories;
|
|
|
|
|
|
struct spdk_trace_histories {
|
|
struct spdk_trace_flags flags;
|
|
|
|
/**
|
|
* struct spdk_trace_history has a dynamic size determined by num_entries
|
|
* in spdk_trace_init. Mark array size of per_lcore_history to be 0 in uint8_t
|
|
* as a reminder that each per_lcore_history pointer should be gotten by
|
|
* proper API, instead of directly referencing by struct element.
|
|
*/
|
|
uint8_t per_lcore_history[0];
|
|
};
|
|
|
|
static inline uint64_t
|
|
spdk_get_trace_history_size(uint64_t num_entries)
|
|
{
|
|
return sizeof(struct spdk_trace_history) + num_entries * sizeof(struct spdk_trace_entry);
|
|
}
|
|
|
|
static inline uint64_t
|
|
spdk_get_trace_histories_size(struct spdk_trace_histories *trace_histories)
|
|
{
|
|
return trace_histories->flags.lcore_history_offsets[SPDK_TRACE_MAX_LCORE];
|
|
}
|
|
|
|
static inline struct spdk_trace_history *
|
|
spdk_get_per_lcore_history(struct spdk_trace_histories *trace_histories, unsigned lcore)
|
|
{
|
|
char *lcore_history_offset;
|
|
|
|
if (lcore >= SPDK_TRACE_MAX_LCORE) {
|
|
return NULL;
|
|
}
|
|
|
|
lcore_history_offset = (char *)trace_histories;
|
|
lcore_history_offset += trace_histories->flags.lcore_history_offsets[lcore];
|
|
|
|
return (struct spdk_trace_history *)lcore_history_offset;
|
|
}
|
|
|
|
void _spdk_trace_record(uint64_t tsc, uint16_t tpoint_id, uint16_t poller_id,
|
|
uint32_t size, uint64_t object_id, int num_args, ...);
|
|
|
|
#define _spdk_trace_record_tsc(tsc, tpoint_id, poller_id, size, object_id, num_args, ...) \
|
|
do { \
|
|
assert(tpoint_id < SPDK_TRACE_MAX_TPOINT_ID); \
|
|
if (g_trace_histories == NULL || \
|
|
!((1ULL << (tpoint_id & 0x3F)) & \
|
|
g_trace_histories->flags.tpoint_mask[tpoint_id >> 6])) { \
|
|
break; \
|
|
} \
|
|
_spdk_trace_record(tsc, tpoint_id, poller_id, size, object_id, \
|
|
num_args, ## __VA_ARGS__); \
|
|
} while (0)
|
|
|
|
/* Return the number of variable arguments. */
|
|
#define spdk_trace_num_args(...) _spdk_trace_num_args(, ## __VA_ARGS__)
|
|
#define _spdk_trace_num_args(...) __spdk_trace_num_args(__VA_ARGS__, 8, 7, 6, 5, 4, 3, 2, 1, 0)
|
|
#define __spdk_trace_num_args(v, a1, a2, a3, a4, a5, a6, a7, a8, count, ...) count
|
|
|
|
/**
|
|
* Record the current trace state for tracing tpoints. Debug tool can read the
|
|
* information from shared memory to post-process the tpoint entries and display
|
|
* in a human-readable format.
|
|
*
|
|
* \param tsc Current tsc.
|
|
* \param tpoint_id Tracepoint id to record.
|
|
* \param poller_id Poller id to record.
|
|
* \param size Size to record.
|
|
* \param object_id Object id to record.
|
|
* \param ... Extra tracepoint arguments. The number, types, and order of the arguments
|
|
* must match the definition of the tracepoint.
|
|
*/
|
|
#define spdk_trace_record_tsc(tsc, tpoint_id, poller_id, size, object_id, ...) \
|
|
_spdk_trace_record_tsc(tsc, tpoint_id, poller_id, size, object_id, \
|
|
spdk_trace_num_args(__VA_ARGS__), ## __VA_ARGS__)
|
|
|
|
/**
|
|
* Record the current trace state for tracing tpoints. Debug tool can read the
|
|
* information from shared memory to post-process the tpoint entries and display
|
|
* in a human-readable format. This macro will call spdk_get_ticks() to get
|
|
* the current tsc to save in the tracepoint.
|
|
*
|
|
* \param tpoint_id Tracepoint id to record.
|
|
* \param poller_id Poller id to record.
|
|
* \param size Size to record.
|
|
* \param object_id Object id to record.
|
|
* \param ... Extra tracepoint arguments. The number, types, and order of the arguments
|
|
* must match the definition of the tracepoint.
|
|
*/
|
|
#define spdk_trace_record(tpoint_id, poller_id, size, object_id, ...) \
|
|
spdk_trace_record_tsc(0, tpoint_id, poller_id, size, object_id, ## __VA_ARGS__)
|
|
|
|
/**
|
|
* Get the current tpoint mask of the given tpoint group.
|
|
*
|
|
* \param group_id Tpoint group id associated with the tpoint mask.
|
|
*
|
|
* \return current tpoint mask.
|
|
*/
|
|
uint64_t spdk_trace_get_tpoint_mask(uint32_t group_id);
|
|
|
|
/**
|
|
* Add the specified tpoints to the current tpoint mask for the given tpoint group.
|
|
*
|
|
* \param group_id Tpoint group id associated with the tpoint mask.
|
|
* \param tpoint_mask Tpoint mask which indicates which tpoints to add to the
|
|
* current tpoint mask.
|
|
*/
|
|
void spdk_trace_set_tpoints(uint32_t group_id, uint64_t tpoint_mask);
|
|
|
|
/**
|
|
* Clear the specified tpoints from the current tpoint mask for the given tpoint group.
|
|
*
|
|
* \param group_id Tpoint group id associated with the tpoint mask.
|
|
* \param tpoint_mask Tpoint mask which indicates which tpoints to clear from
|
|
* the current tpoint mask.
|
|
*/
|
|
void spdk_trace_clear_tpoints(uint32_t group_id, uint64_t tpoint_mask);
|
|
|
|
/**
|
|
* Get a mask of all tracepoint groups which have at least one tracepoint enabled.
|
|
*
|
|
* \return a mask of all tracepoint groups.
|
|
*/
|
|
uint64_t spdk_trace_get_tpoint_group_mask(void);
|
|
|
|
/**
|
|
* For each tpoint group specified in the group mask, enable all of its tpoints.
|
|
*
|
|
* \param tpoint_group_mask Tpoint group mask that indicates which tpoints to enable.
|
|
*/
|
|
void spdk_trace_set_tpoint_group_mask(uint64_t tpoint_group_mask);
|
|
|
|
/**
|
|
* For each tpoint group specified in the group mask, disable all of its tpoints.
|
|
*
|
|
* \param tpoint_group_mask Tpoint group mask that indicates which tpoints to disable.
|
|
*/
|
|
void spdk_trace_clear_tpoint_group_mask(uint64_t tpoint_group_mask);
|
|
|
|
/**
|
|
* Initialize the trace environment. Debug tool can read the information from
|
|
* the given shared memory to post-process the tpoint entries and display in a
|
|
* human-readable format.
|
|
*
|
|
* \param shm_name Name of shared memory.
|
|
* \param num_entries Number of trace entries per lcore.
|
|
* \return 0 on success, else non-zero indicates a failure.
|
|
*/
|
|
int spdk_trace_init(const char *shm_name, uint64_t num_entries);
|
|
|
|
/**
|
|
* Unmap global trace memory structs.
|
|
*/
|
|
void spdk_trace_cleanup(void);
|
|
|
|
/**
|
|
* Initialize trace flags.
|
|
*/
|
|
void spdk_trace_flags_init(void);
|
|
|
|
#define OWNER_NONE 0
|
|
#define OBJECT_NONE 0
|
|
|
|
/**
|
|
* Register the trace owner.
|
|
*
|
|
* \param type Type of the trace owner.
|
|
* \param id_prefix Prefix of id for the trace owner.
|
|
*/
|
|
void spdk_trace_register_owner(uint8_t type, char id_prefix);
|
|
|
|
/**
|
|
* Register the trace object.
|
|
*
|
|
* \param type Type of the trace object.
|
|
* \param id_prefix Prefix of id for the trace object.
|
|
*/
|
|
void spdk_trace_register_object(uint8_t type, char id_prefix);
|
|
|
|
/**
|
|
* Register the description for a tpoint with a single argument.
|
|
*
|
|
* \param name Name for the tpoint.
|
|
* \param tpoint_id Id for the tpoint.
|
|
* \param owner_type Owner type for the tpoint.
|
|
* \param object_type Object type for the tpoint.
|
|
* \param new_object New object for the tpoint.
|
|
* \param arg1_type Type of arg1.
|
|
* \param arg1_name Name of argument.
|
|
*/
|
|
void spdk_trace_register_description(const char *name, uint16_t tpoint_id, uint8_t owner_type,
|
|
uint8_t object_type, uint8_t new_object,
|
|
uint8_t arg1_type, const char *arg1_name);
|
|
|
|
struct spdk_trace_tpoint_opts {
|
|
const char *name;
|
|
uint16_t tpoint_id;
|
|
uint8_t owner_type;
|
|
uint8_t object_type;
|
|
uint8_t new_object;
|
|
struct {
|
|
const char *name;
|
|
uint8_t type;
|
|
uint8_t size;
|
|
} args[SPDK_TRACE_MAX_ARGS_COUNT];
|
|
};
|
|
|
|
/**
|
|
* Register the description for a number of tpoints. This function allows the user to register
|
|
* tracepoints with multiple arguments (up to 5).
|
|
*
|
|
* \param opts Array of structures describing tpoints and their arguments.
|
|
* \param num_opts Number of tpoints to register (size of the opts array).
|
|
*/
|
|
void spdk_trace_register_description_ext(const struct spdk_trace_tpoint_opts *opts,
|
|
size_t num_opts);
|
|
|
|
struct spdk_trace_register_fn *spdk_trace_get_first_register_fn(void);
|
|
|
|
struct spdk_trace_register_fn *spdk_trace_get_next_register_fn(struct spdk_trace_register_fn
|
|
*register_fn);
|
|
|
|
/**
|
|
* Enable trace on specific tpoint group
|
|
*
|
|
* \param group_name Name of group to enable, "all" for enabling all groups.
|
|
* \return 0 on success, else non-zero indicates a failure.
|
|
*/
|
|
int spdk_trace_enable_tpoint_group(const char *group_name);
|
|
|
|
/**
|
|
* Disable trace on specific tpoint group
|
|
*
|
|
* \param group_name Name of group to disable, "all" for disabling all groups.
|
|
* \return 0 on success, else non-zero indicates a failure.
|
|
*/
|
|
int spdk_trace_disable_tpoint_group(const char *group_name);
|
|
|
|
/**
|
|
* Show trace mask and its usage.
|
|
*
|
|
* \param f File to hold the mask's information.
|
|
* \param tmask_arg Command line option to set the trace group mask.
|
|
*/
|
|
void spdk_trace_mask_usage(FILE *f, const char *tmask_arg);
|
|
|
|
struct spdk_trace_register_fn {
|
|
const char *name;
|
|
uint8_t tgroup_id;
|
|
void (*reg_fn)(void);
|
|
struct spdk_trace_register_fn *next;
|
|
};
|
|
|
|
/**
|
|
* Add new trace register function.
|
|
*
|
|
* \param reg_fn Trace register function to add.
|
|
*/
|
|
void spdk_trace_add_register_fn(struct spdk_trace_register_fn *reg_fn);
|
|
|
|
#define SPDK_TRACE_REGISTER_FN(fn, name_str, _tgroup_id) \
|
|
static void fn(void); \
|
|
struct spdk_trace_register_fn reg_ ## fn = { \
|
|
.name = name_str, \
|
|
.tgroup_id = _tgroup_id, \
|
|
.reg_fn = fn, \
|
|
.next = NULL, \
|
|
}; \
|
|
__attribute__((constructor)) static void _ ## fn(void) \
|
|
{ \
|
|
spdk_trace_add_register_fn(®_ ## fn); \
|
|
} \
|
|
static void fn(void)
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif
|