add profiler
This commit is contained in:
34
src/profiler/profiler.c
Normal file
34
src/profiler/profiler.c
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#include "profiler_spall.c"
|
||||||
|
|
||||||
|
static SpallProfile spall_ctx;
|
||||||
|
static SpallBuffer spall_buffer;
|
||||||
|
|
||||||
|
fn void profiler__begin() {
|
||||||
|
spall_ctx = spall_init_file("hello_world.spall", 1);
|
||||||
|
|
||||||
|
int buffer_size = 1 * 1024 * 1024;
|
||||||
|
void *buffer = ma_push_size(&tcx->perm, buffer_size);
|
||||||
|
spall_buffer = (SpallBuffer){buffer, (size_t)buffer_size};
|
||||||
|
|
||||||
|
spall_buffer_init(&spall_ctx, &spall_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn void profiler__end() {
|
||||||
|
spall_buffer_quit(&spall_ctx, &spall_buffer);
|
||||||
|
free(spall_buffer.data);
|
||||||
|
spall_quit(&spall_ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn void profiler__begin_scope(const char *name, int len) {
|
||||||
|
spall_buffer_begin(&spall_ctx, &spall_buffer,
|
||||||
|
name, // name of your name
|
||||||
|
len, // name len minus the null terminator
|
||||||
|
os_microseconds() // timestamp in microseconds -- start of your timing block
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn void profiler__end_scope() {
|
||||||
|
spall_buffer_end(&spall_ctx, &spall_buffer,
|
||||||
|
os_microseconds() // timestamp in microseconds -- end of your timing block
|
||||||
|
);
|
||||||
|
}
|
||||||
22
src/profiler/profiler.h
Normal file
22
src/profiler/profiler.h
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
#ifndef PROFILER_ENABLED
|
||||||
|
#define PROFILER_ENABLED 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if PROFILER_ENABLED
|
||||||
|
#define profiler_begin() profiler__begin()
|
||||||
|
#define profiler_end() profiler__end()
|
||||||
|
#define profiler_begin_scope(name) profiler__begin_scope(#name, sizeof(#name) - 1)
|
||||||
|
#define profiler_begin_fn() profiler_begin_scope(#__FUNCTION__)
|
||||||
|
#define profiler_end_scope() profiler__end_scope()
|
||||||
|
|
||||||
|
fn void profiler__begin();
|
||||||
|
fn void profiler__end();
|
||||||
|
fn void profiler__begin_scope(const char *name, int len);
|
||||||
|
fn void profiler__end_scope();
|
||||||
|
#else
|
||||||
|
#define profiler_begin()
|
||||||
|
#define profiler_end()
|
||||||
|
#define profiler_begin_scope(name)
|
||||||
|
#define profiler_begin_fn()
|
||||||
|
#define profiler_end_scope()
|
||||||
|
#endif
|
||||||
443
src/profiler/profiler_spall.c
Normal file
443
src/profiler/profiler_spall.c
Normal file
@@ -0,0 +1,443 @@
|
|||||||
|
// SPDX-FileCopyrightText: © 2023 Phillip Trudeau-Tavara <pmttavara@protonmail.com>
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
TODO: Optional Helper APIs:
|
||||||
|
|
||||||
|
- Compression API: would require a mutexed lockable context (yuck...)
|
||||||
|
- Either using a ZIP library, a name cache + TIDPID cache, or both (but ZIP is likely more than enough!!!)
|
||||||
|
- begin()/end() writes compressed chunks to a caller-determined destination
|
||||||
|
- The destination can be the buffered-writing API or a custom user destination
|
||||||
|
- Ultimately need to take a lock with some granularity... can that be the caller's responsibility?
|
||||||
|
|
||||||
|
- Counter Event: should allow tracking arbitrary named values with a single event, for memory and frame profiling
|
||||||
|
|
||||||
|
- Ring-buffer API
|
||||||
|
spall_ring_init
|
||||||
|
spall_ring_emit_begin
|
||||||
|
spall_ring_emit_end
|
||||||
|
spall_ring_flush
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef SPALL_H
|
||||||
|
#define SPALL_H
|
||||||
|
|
||||||
|
#if !defined(_MSC_VER) || defined(__clang__)
|
||||||
|
#define SPALL_NOINSTRUMENT __attribute__((no_instrument_function))
|
||||||
|
#define SPALL_FORCEINLINE __attribute__((always_inline))
|
||||||
|
#else
|
||||||
|
#define SPALL_NOINSTRUMENT // Can't noinstrument on MSVC!
|
||||||
|
#define SPALL_FORCEINLINE __forceinline
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// #include <stdint.h>
|
||||||
|
// #include <stdio.h>
|
||||||
|
// #include <string.h>
|
||||||
|
// #include <stdb32.h>
|
||||||
|
|
||||||
|
#define SPALL_FN static inline SPALL_NOINSTRUMENT
|
||||||
|
|
||||||
|
#define SPALL_MIN(a, b) (((a) < (b)) ? (a) : (b))
|
||||||
|
|
||||||
|
#pragma pack(push, 1)
|
||||||
|
|
||||||
|
typedef struct SpallHeader {
|
||||||
|
uint64_t magic_header; // = 0x0BADF00D
|
||||||
|
uint64_t version; // = 1
|
||||||
|
double timestamp_unit;
|
||||||
|
uint64_t must_be_0;
|
||||||
|
} SpallHeader;
|
||||||
|
|
||||||
|
enum {
|
||||||
|
SpallEventType_Invalid = 0,
|
||||||
|
SpallEventType_Custom_Data = 1, // Basic readers can skip this.
|
||||||
|
SpallEventType_StreamOver = 2,
|
||||||
|
|
||||||
|
SpallEventType_Begin = 3,
|
||||||
|
SpallEventType_End = 4,
|
||||||
|
SpallEventType_Instant = 5,
|
||||||
|
|
||||||
|
SpallEventType_Overwrite_Timestamp = 6, // Retroactively change timestamp units - useful for incrementally improving RDTSC frequency.
|
||||||
|
SpallEventType_Pad_Skip = 7,
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct SpallBeginEvent {
|
||||||
|
uint8_t type; // = SpallEventType_Begin
|
||||||
|
uint8_t category;
|
||||||
|
|
||||||
|
uint32_t pid;
|
||||||
|
uint32_t tid;
|
||||||
|
double when;
|
||||||
|
|
||||||
|
uint8_t name_length;
|
||||||
|
uint8_t args_length;
|
||||||
|
} SpallBeginEvent;
|
||||||
|
|
||||||
|
typedef struct SpallBeginEventMax {
|
||||||
|
SpallBeginEvent event;
|
||||||
|
char name_bytes[255];
|
||||||
|
char args_bytes[255];
|
||||||
|
} SpallBeginEventMax;
|
||||||
|
|
||||||
|
typedef struct SpallEndEvent {
|
||||||
|
uint8_t type; // = SpallEventType_End
|
||||||
|
uint32_t pid;
|
||||||
|
uint32_t tid;
|
||||||
|
double when;
|
||||||
|
} SpallEndEvent;
|
||||||
|
|
||||||
|
typedef struct SpallPadSkipEvent {
|
||||||
|
uint8_t type; // = SpallEventType_Pad_Skip
|
||||||
|
uint32_t size;
|
||||||
|
} SpallPadSkipEvent;
|
||||||
|
|
||||||
|
#pragma pack(pop)
|
||||||
|
|
||||||
|
typedef struct SpallProfile SpallProfile;
|
||||||
|
|
||||||
|
// Important!: If you define your own callbacks, mark them SPALL_NOINSTRUMENT!
|
||||||
|
typedef b32 (*SpallWriteCallback)(SpallProfile *self, const void *data, size_t length);
|
||||||
|
typedef b32 (*SpallFlushCallback)(SpallProfile *self);
|
||||||
|
typedef void (*SpallCloseCallback)(SpallProfile *self);
|
||||||
|
|
||||||
|
struct SpallProfile {
|
||||||
|
double timestamp_unit;
|
||||||
|
b32 is_json;
|
||||||
|
SpallWriteCallback write;
|
||||||
|
SpallFlushCallback flush;
|
||||||
|
SpallCloseCallback close;
|
||||||
|
void *data;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Important!: If you are writing Begin/End events, then do NOT write
|
||||||
|
// events for the same PID + TID pair on different buffers!!!
|
||||||
|
typedef struct SpallBuffer {
|
||||||
|
void *data;
|
||||||
|
size_t length;
|
||||||
|
|
||||||
|
// Internal data - don't assign this
|
||||||
|
size_t head;
|
||||||
|
SpallProfile *ctx;
|
||||||
|
} SpallBuffer;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(SPALL_BUFFER_PROFILING) && !defined(SPALL_BUFFER_PROFILING_GET_TIME)
|
||||||
|
#error "You must #define SPALL_BUFFER_PROFILING_GET_TIME() to profile buffer flushes."
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE void spall__buffer_profile(SpallProfile *ctx, SpallBuffer *wb, double spall_time_begin, double spall_time_end, const char *name, int name_len);
|
||||||
|
#ifdef SPALL_BUFFER_PROFILING
|
||||||
|
#define SPALL_BUFFER_PROFILE_BEGIN() double spall_time_begin = (SPALL_BUFFER_PROFILING_GET_TIME())
|
||||||
|
// Don't call this with anything other than a string literal
|
||||||
|
#define SPALL_BUFFER_PROFILE_END(name) spall__buffer_profile(ctx, wb, spall_time_begin, (SPALL_BUFFER_PROFILING_GET_TIME()), "" name "", sizeof("" name "") - 1)
|
||||||
|
#else
|
||||||
|
#define SPALL_BUFFER_PROFILE_BEGIN()
|
||||||
|
#define SPALL_BUFFER_PROFILE_END(name)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall__file_write(SpallProfile *ctx, const void *p, size_t n) {
|
||||||
|
if (!ctx->data) return false;
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (feof((FILE *)ctx->data)) return false;
|
||||||
|
if (ferror((FILE *)ctx->data)) return false;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (fwrite(p, n, 1, (FILE *)ctx->data) != 1) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
SPALL_FN b32 spall__file_flush(SpallProfile *ctx) {
|
||||||
|
if (!ctx->data) return false;
|
||||||
|
if (fflush((FILE *)ctx->data)) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
SPALL_FN void spall__file_close(SpallProfile *ctx) {
|
||||||
|
if (!ctx->data) return;
|
||||||
|
|
||||||
|
if (ctx->is_json) {
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (!feof((FILE *)ctx->data) && !ferror((FILE *)ctx->data))
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
fseek((FILE *)ctx->data, -2, SEEK_CUR); // seek back to overwrite trailing comma
|
||||||
|
fwrite("\n]}\n", sizeof("\n]}\n") - 1, 1, (FILE *)ctx->data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fflush((FILE *)ctx->data);
|
||||||
|
fclose((FILE *)ctx->data);
|
||||||
|
ctx->data = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall__buffer_flush(SpallProfile *ctx, SpallBuffer *wb) {
|
||||||
|
// precon: wb
|
||||||
|
// precon: wb->data
|
||||||
|
// precon: wb->head <= wb->length
|
||||||
|
// precon: !ctx || ctx->write
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (wb->ctx != ctx) return false; // Buffer must be bound to this context (or to NULL)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (wb->head && ctx) {
|
||||||
|
SPALL_BUFFER_PROFILE_BEGIN();
|
||||||
|
if (!ctx->write) return false;
|
||||||
|
if (ctx->write == spall__file_write) {
|
||||||
|
if (!spall__file_write(ctx, wb->data, wb->head)) return false;
|
||||||
|
} else {
|
||||||
|
if (!ctx->write(ctx, wb->data, wb->head)) return false;
|
||||||
|
}
|
||||||
|
SPALL_BUFFER_PROFILE_END("Buffer Flush");
|
||||||
|
}
|
||||||
|
wb->head = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall__buffer_write(SpallProfile *ctx, SpallBuffer *wb, void *p, size_t n) {
|
||||||
|
// precon: !wb || wb->head < wb->length
|
||||||
|
// precon: !ctx || ctx->write
|
||||||
|
if (!wb) return ctx->write && ctx->write(ctx, p, n);
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (wb->ctx != ctx) return false; // Buffer must be bound to this context (or to NULL)
|
||||||
|
#endif
|
||||||
|
if (wb->head + n > wb->length && !spall__buffer_flush(ctx, wb)) return false;
|
||||||
|
if (n > wb->length) {
|
||||||
|
SPALL_BUFFER_PROFILE_BEGIN();
|
||||||
|
if (!ctx->write || !ctx->write(ctx, p, n)) return false;
|
||||||
|
SPALL_BUFFER_PROFILE_END("Unbuffered Write");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
memcpy((char *)wb->data + wb->head, p, n);
|
||||||
|
wb->head += n;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_buffer_flush(SpallProfile *ctx, SpallBuffer *wb) {
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (!wb) return false;
|
||||||
|
if (!wb->data) return false;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!spall__buffer_flush(ctx, wb)) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_buffer_init(SpallProfile *ctx, SpallBuffer *wb) {
|
||||||
|
if (!spall_buffer_flush(NULL, wb)) return false;
|
||||||
|
wb->ctx = ctx;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
SPALL_FN b32 spall_buffer_quit(SpallProfile *ctx, SpallBuffer *wb) {
|
||||||
|
if (!spall_buffer_flush(ctx, wb)) return false;
|
||||||
|
wb->ctx = NULL;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_buffer_abort(SpallBuffer *wb) {
|
||||||
|
if (!wb) return false;
|
||||||
|
wb->ctx = NULL;
|
||||||
|
if (!spall__buffer_flush(NULL, wb)) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN size_t spall_build_header(void *buffer, size_t rem_size, double timestamp_unit) {
|
||||||
|
size_t header_size = sizeof(SpallHeader);
|
||||||
|
if (header_size > rem_size) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SpallHeader *header = (SpallHeader *)buffer;
|
||||||
|
header->magic_header = 0x0BADF00D;
|
||||||
|
header->version = 1;
|
||||||
|
header->timestamp_unit = timestamp_unit;
|
||||||
|
header->must_be_0 = 0;
|
||||||
|
return header_size;
|
||||||
|
}
|
||||||
|
SPALL_FN SPALL_FORCEINLINE size_t spall_build_begin(void *buffer, size_t rem_size, const char *name, signed long name_len, const char *args, signed long args_len, double when, uint32_t tid, uint32_t pid) {
|
||||||
|
SpallBeginEventMax *ev = (SpallBeginEventMax *)buffer;
|
||||||
|
uint8_t trunc_name_len = (uint8_t)SPALL_MIN(name_len, 255); // will be interpreted as truncated in the app (?)
|
||||||
|
uint8_t trunc_args_len = (uint8_t)SPALL_MIN(args_len, 255); // will be interpreted as truncated in the app (?)
|
||||||
|
|
||||||
|
size_t ev_size = sizeof(SpallBeginEvent) + trunc_name_len + trunc_args_len;
|
||||||
|
if (ev_size > rem_size) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ev->event.type = SpallEventType_Begin;
|
||||||
|
ev->event.category = 0;
|
||||||
|
ev->event.pid = pid;
|
||||||
|
ev->event.tid = tid;
|
||||||
|
ev->event.when = when;
|
||||||
|
ev->event.name_length = trunc_name_len;
|
||||||
|
ev->event.args_length = trunc_args_len;
|
||||||
|
memcpy(ev->name_bytes, name, trunc_name_len);
|
||||||
|
memcpy(ev->name_bytes + trunc_name_len, args, trunc_args_len);
|
||||||
|
|
||||||
|
return ev_size;
|
||||||
|
}
|
||||||
|
SPALL_FN SPALL_FORCEINLINE size_t spall_build_end(void *buffer, size_t rem_size, double when, uint32_t tid, uint32_t pid) {
|
||||||
|
size_t ev_size = sizeof(SpallEndEvent);
|
||||||
|
if (ev_size > rem_size) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SpallEndEvent *ev = (SpallEndEvent *)buffer;
|
||||||
|
ev->type = SpallEventType_End;
|
||||||
|
ev->pid = pid;
|
||||||
|
ev->tid = tid;
|
||||||
|
ev->when = when;
|
||||||
|
|
||||||
|
return ev_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN void spall_quit(SpallProfile *ctx) {
|
||||||
|
if (!ctx) return;
|
||||||
|
if (ctx->close) ctx->close(ctx);
|
||||||
|
|
||||||
|
memset(ctx, 0, sizeof(*ctx));
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SpallProfile spall_init_callbacks(double timestamp_unit,
|
||||||
|
SpallWriteCallback write,
|
||||||
|
SpallFlushCallback flush,
|
||||||
|
SpallCloseCallback close,
|
||||||
|
void *userdata,
|
||||||
|
b32 is_json) {
|
||||||
|
SpallProfile ctx;
|
||||||
|
memset(&ctx, 0, sizeof(ctx));
|
||||||
|
if (timestamp_unit < 0) return ctx;
|
||||||
|
ctx.timestamp_unit = timestamp_unit;
|
||||||
|
ctx.is_json = is_json;
|
||||||
|
ctx.data = userdata;
|
||||||
|
ctx.write = write;
|
||||||
|
ctx.flush = flush;
|
||||||
|
ctx.close = close;
|
||||||
|
|
||||||
|
if (ctx.is_json) {
|
||||||
|
if (!ctx.write(&ctx, "{\"traceEvents\":[\n", sizeof("{\"traceEvents\":[\n") - 1)) {
|
||||||
|
spall_quit(&ctx);
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
SpallHeader header;
|
||||||
|
size_t len = spall_build_header(&header, sizeof(header), timestamp_unit);
|
||||||
|
if (!ctx.write(&ctx, &header, len)) {
|
||||||
|
spall_quit(&ctx);
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SpallProfile spall_init_file_ex(const char *filename, double timestamp_unit, b32 is_json) {
|
||||||
|
SpallProfile ctx;
|
||||||
|
memset(&ctx, 0, sizeof(ctx));
|
||||||
|
if (!filename) return ctx;
|
||||||
|
ctx.data = fopen(filename, "wb"); // TODO: handle utf8 and long paths on windows
|
||||||
|
if (ctx.data) { // basically freopen() but we don't want to force users to lug along another macro define
|
||||||
|
fclose((FILE *)ctx.data);
|
||||||
|
ctx.data = fopen(filename, "ab");
|
||||||
|
}
|
||||||
|
if (!ctx.data) {
|
||||||
|
spall_quit(&ctx);
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
ctx = spall_init_callbacks(timestamp_unit, spall__file_write, spall__file_flush, spall__file_close, ctx.data, is_json);
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SpallProfile spall_init_file(const char *filename, double timestamp_unit) { return spall_init_file_ex(filename, timestamp_unit, false); }
|
||||||
|
SPALL_FN SpallProfile spall_init_file_json(const char *filename, double timestamp_unit) { return spall_init_file_ex(filename, timestamp_unit, true); }
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_flush(SpallProfile *ctx) {
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (!ctx) return false;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!ctx->flush || !ctx->flush(ctx)) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall_buffer_begin_args(SpallProfile *ctx, SpallBuffer *wb, const char *name, signed long name_len, const char *args, signed long args_len, double when, uint32_t tid, uint32_t pid) {
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (!ctx) return false;
|
||||||
|
if (!name) return false;
|
||||||
|
if (name_len <= 0) return false;
|
||||||
|
if (!wb) return false;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (ctx->is_json) {
|
||||||
|
char buf[1024];
|
||||||
|
int buf_len = snprintf(buf, sizeof(buf),
|
||||||
|
"{\"ph\":\"B\",\"ts\":%f,\"pid\":%u,\"tid\":%u,\"name\":\"%.*s\",\"args\":\"%.*s\"},\n",
|
||||||
|
when * ctx->timestamp_unit, pid, tid, (int)(uint8_t)name_len, name, (int)(uint8_t)args_len, args);
|
||||||
|
if (buf_len <= 0) return false;
|
||||||
|
if (buf_len >= sizeof(buf)) return false;
|
||||||
|
if (!spall__buffer_write(ctx, wb, buf, buf_len)) return false;
|
||||||
|
} else {
|
||||||
|
if ((wb->head + sizeof(SpallBeginEventMax)) > wb->length) {
|
||||||
|
if (!spall__buffer_flush(ctx, wb)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wb->head += spall_build_begin((char *)wb->data + wb->head, wb->length - wb->head, name, name_len, args, args_len, when, tid, pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall_buffer_begin_ex(SpallProfile *ctx, SpallBuffer *wb, const char *name, signed long name_len, double when, uint32_t tid, uint32_t pid) {
|
||||||
|
return spall_buffer_begin_args(ctx, wb, name, name_len, "", 0, when, tid, pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_buffer_begin(SpallProfile *ctx, SpallBuffer *wb, const char *name, signed long name_len, double when) {
|
||||||
|
return spall_buffer_begin_args(ctx, wb, name, name_len, "", 0, when, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE b32 spall_buffer_end_ex(SpallProfile *ctx, SpallBuffer *wb, double when, uint32_t tid, uint32_t pid) {
|
||||||
|
#ifdef SPALL_DEBUG
|
||||||
|
if (!ctx) return false;
|
||||||
|
if (!wb) return false;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (ctx->is_json) {
|
||||||
|
char buf[512];
|
||||||
|
int buf_len = snprintf(buf, sizeof(buf),
|
||||||
|
"{\"ph\":\"E\",\"ts\":%f,\"pid\":%u,\"tid\":%u},\n",
|
||||||
|
when * ctx->timestamp_unit, pid, tid);
|
||||||
|
if (buf_len <= 0) return false;
|
||||||
|
if (buf_len >= sizeof(buf)) return false;
|
||||||
|
if (!spall__buffer_write(ctx, wb, buf, buf_len)) return false;
|
||||||
|
} else {
|
||||||
|
if ((wb->head + sizeof(SpallEndEvent)) > wb->length) {
|
||||||
|
if (!spall__buffer_flush(ctx, wb)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wb->head += spall_build_end((char *)wb->data + wb->head, wb->length - wb->head, when, tid, pid);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
SPALL_FN b32 spall_buffer_end(SpallProfile *ctx, SpallBuffer *wb, double when) { return spall_buffer_end_ex(ctx, wb, when, 0, 0); }
|
||||||
|
|
||||||
|
SPALL_FN SPALL_FORCEINLINE void spall__buffer_profile(SpallProfile *ctx, SpallBuffer *wb, double spall_time_begin, double spall_time_end, const char *name, int name_len) {
|
||||||
|
// precon: ctx
|
||||||
|
// precon: ctx->write
|
||||||
|
char temp_buffer_data[2048];
|
||||||
|
SpallBuffer temp_buffer = {temp_buffer_data, sizeof(temp_buffer_data)};
|
||||||
|
if (!spall_buffer_begin_ex(ctx, &temp_buffer, name, name_len, spall_time_begin, (uint32_t)(uintptr_t)wb->data, 4222222222)) return;
|
||||||
|
if (!spall_buffer_end_ex(ctx, &temp_buffer, spall_time_end, (uint32_t)(uintptr_t)wb->data, 4222222222)) return;
|
||||||
|
if (ctx->write) ctx->write(ctx, temp_buffer_data, temp_buffer.head);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif // SPALL_H
|
||||||
Reference in New Issue
Block a user