compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
/*
|
|
|
|
* Copyright © 2014 Pekka Paalanen <pq@iki.fi>
|
|
|
|
* Copyright © 2014 Collabora, Ltd.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
* a copy of this software and associated documentation files (the
|
|
|
|
* "Software"), to deal in the Software without restriction, including
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
* the following conditions:
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice (including the
|
|
|
|
* next paragraph) shall be included in all copies or substantial
|
|
|
|
* portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "timeline.h"
|
|
|
|
#include "compositor.h"
|
|
|
|
#include "file-util.h"
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
|
|
|
|
struct timeline_log {
|
|
|
|
clock_t clk_id;
|
|
|
|
FILE *file;
|
|
|
|
unsigned series;
|
|
|
|
struct wl_listener compositor_destroy_listener;
|
|
|
|
};
|
|
|
|
|
|
|
|
WL_EXPORT int weston_timeline_enabled_;
|
|
|
|
static struct timeline_log timeline_ = { CLOCK_MONOTONIC, NULL, 0 };
|
|
|
|
|
|
|
|
static int
|
|
|
|
weston_timeline_do_open(void)
|
|
|
|
{
|
|
|
|
const char *prefix = "weston-timeline-";
|
|
|
|
const char *suffix = ".log";
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
char fname[1000];
|
|
|
|
|
|
|
|
timeline_.file = file_create_dated(NULL, prefix, suffix,
|
|
|
|
fname, sizeof(fname));
|
|
|
|
if (!timeline_.file) {
|
|
|
|
const char *msg;
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
|
|
|
|
switch (errno) {
|
|
|
|
case ETIME:
|
|
|
|
msg = "failure in datetime formatting";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
msg = strerror(errno);
|
|
|
|
}
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
|
|
|
|
weston_log("Cannot open '%s*%s' for writing: %s\n",
|
|
|
|
prefix, suffix, msg);
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
weston_log("Opened timeline file '%s'\n", fname);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
timeline_notify_destroy(struct wl_listener *listener, void *data)
|
|
|
|
{
|
|
|
|
weston_timeline_close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
weston_timeline_open(struct weston_compositor *compositor)
|
|
|
|
{
|
|
|
|
if (weston_timeline_enabled_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (weston_timeline_do_open() < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
timeline_.compositor_destroy_listener.notify = timeline_notify_destroy;
|
|
|
|
wl_signal_add(&compositor->destroy_signal,
|
|
|
|
&timeline_.compositor_destroy_listener);
|
|
|
|
|
|
|
|
if (++timeline_.series == 0)
|
|
|
|
++timeline_.series;
|
|
|
|
|
|
|
|
weston_timeline_enabled_ = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
weston_timeline_close(void)
|
|
|
|
{
|
|
|
|
if (!weston_timeline_enabled_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
weston_timeline_enabled_ = 0;
|
|
|
|
|
|
|
|
wl_list_remove(&timeline_.compositor_destroy_listener.link);
|
|
|
|
|
|
|
|
fclose(timeline_.file);
|
|
|
|
timeline_.file = NULL;
|
|
|
|
weston_log("Timeline log file closed.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct timeline_emit_context {
|
|
|
|
FILE *cur;
|
|
|
|
FILE *out;
|
|
|
|
unsigned series;
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned
|
|
|
|
timeline_new_id(void)
|
|
|
|
{
|
|
|
|
static unsigned idc;
|
|
|
|
|
|
|
|
if (++idc == 0)
|
|
|
|
++idc;
|
|
|
|
|
|
|
|
return idc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
check_series(struct timeline_emit_context *ctx,
|
|
|
|
struct weston_timeline_object *to)
|
|
|
|
{
|
|
|
|
if (to->series == 0 || to->series != ctx->series) {
|
|
|
|
to->series = ctx->series;
|
|
|
|
to->id = timeline_new_id();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (to->force_refresh) {
|
|
|
|
to->force_refresh = 0;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
fprint_quoted_string(FILE *fp, const char *str)
|
|
|
|
{
|
|
|
|
if (!str) {
|
|
|
|
fprintf(fp, "null");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(fp, "\"%s\"", str);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
emit_weston_output(struct timeline_emit_context *ctx, void *obj)
|
|
|
|
{
|
|
|
|
struct weston_output *o = obj;
|
|
|
|
|
|
|
|
if (check_series(ctx, &o->timeline)) {
|
|
|
|
fprintf(ctx->out, "{ \"id\":%u, "
|
|
|
|
"\"type\":\"weston_output\", \"name\":",
|
|
|
|
o->timeline.id);
|
|
|
|
fprint_quoted_string(ctx->out, o->name);
|
|
|
|
fprintf(ctx->out, " }\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(ctx->cur, "\"wo\":%u", o->timeline.id);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
check_weston_surface_description(struct timeline_emit_context *ctx,
|
|
|
|
struct weston_surface *s)
|
|
|
|
{
|
|
|
|
struct weston_surface *mains;
|
|
|
|
char d[512];
|
|
|
|
char mainstr[32];
|
|
|
|
|
|
|
|
if (!check_series(ctx, &s->timeline))
|
|
|
|
return;
|
|
|
|
|
|
|
|
mains = weston_surface_get_main_surface(s);
|
|
|
|
if (mains != s) {
|
|
|
|
check_weston_surface_description(ctx, mains);
|
|
|
|
if (snprintf(mainstr, sizeof(mainstr),
|
|
|
|
", \"main_surface\":%u", mains->timeline.id) < 0)
|
|
|
|
mainstr[0] = '\0';
|
|
|
|
} else {
|
|
|
|
mainstr[0] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!s->get_label || s->get_label(s, d, sizeof(d)) < 0)
|
|
|
|
d[0] = '\0';
|
|
|
|
|
|
|
|
fprintf(ctx->out, "{ \"id\":%u, "
|
|
|
|
"\"type\":\"weston_surface\", \"desc\":", s->timeline.id);
|
|
|
|
fprint_quoted_string(ctx->out, d[0] ? d : NULL);
|
|
|
|
fprintf(ctx->out, "%s }\n", mainstr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
emit_weston_surface(struct timeline_emit_context *ctx, void *obj)
|
|
|
|
{
|
|
|
|
struct weston_surface *s = obj;
|
|
|
|
|
|
|
|
check_weston_surface_description(ctx, s);
|
|
|
|
fprintf(ctx->cur, "\"ws\":%u", s->timeline.id);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
emit_vblank_timestamp(struct timeline_emit_context *ctx, void *obj)
|
|
|
|
{
|
|
|
|
struct timespec *ts = obj;
|
|
|
|
|
|
|
|
fprintf(ctx->cur, "\"vblank\":[%" PRId64 ", %ld]",
|
|
|
|
(int64_t)ts->tv_sec, ts->tv_nsec);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
emit_gpu_timestamp(struct timeline_emit_context *ctx, void *obj)
|
|
|
|
{
|
|
|
|
struct timespec *ts = obj;
|
|
|
|
|
|
|
|
fprintf(ctx->cur, "\"gpu\":[%" PRId64 ", %ld]",
|
|
|
|
(int64_t)ts->tv_sec, ts->tv_nsec);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
typedef int (*type_func)(struct timeline_emit_context *ctx, void *obj);
|
|
|
|
|
|
|
|
static const type_func type_dispatch[] = {
|
|
|
|
[TLT_OUTPUT] = emit_weston_output,
|
|
|
|
[TLT_SURFACE] = emit_weston_surface,
|
|
|
|
[TLT_VBLANK] = emit_vblank_timestamp,
|
|
|
|
[TLT_GPU] = emit_gpu_timestamp,
|
compositor: Implement JSON-timeline logging
Logging is activated and deactivated with the debug key binding 't'.
When activated, it creates a new log file, where it records the events.
The log file contains events and detailed object information entries in
JSON format, and is meant to be parsed in sequence from beginning to the
end.
The emitted events are mostly related to the output repaint cycle, like
when repaint begins, is submitted to GPU, and when it completes on a
vblank. This is recorded per-output. Also some per-surface events are
recorded, including when surface damage is flushed.
To reduce the log size, events refer to objects like outputs and
surfaces by id numbers. Detailed object information is emitted only as
needed: on the first object occurrence, and afterwards only if
weston_timeline_object::force_refresh asks for it.
The detailed information for surfaces includes the string returned by
weston_surface::get_label. Therefore it is important to set
weston_timeline_object::force_refresh = 1 whenever the string would
change, so that the new details get recorded.
A rudimentary parser and SVG generator can be found at:
https://github.com/ppaalanen/wesgr
The timeline logs can answer questions including:
- How does the compositor repaint cycle work timing-wise?
- When was the vblank deadline missed?
- What is the latency from surface commit to showing the new content on
screen?
- How long does it take to process the scenegraph?
v2: weston_surface::get_description renamed to get_label.
v3: reafctor a bit into fprint_quoted_string().
Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
10 years ago
|
|
|
};
|
|
|
|
|
|
|
|
WL_EXPORT void
|
|
|
|
weston_timeline_point(const char *name, ...)
|
|
|
|
{
|
|
|
|
va_list argp;
|
|
|
|
struct timespec ts;
|
|
|
|
enum timeline_type otype;
|
|
|
|
void *obj;
|
|
|
|
char buf[512];
|
|
|
|
struct timeline_emit_context ctx;
|
|
|
|
|
|
|
|
clock_gettime(timeline_.clk_id, &ts);
|
|
|
|
|
|
|
|
ctx.out = timeline_.file;
|
|
|
|
ctx.cur = fmemopen(buf, sizeof(buf), "w");
|
|
|
|
ctx.series = timeline_.series;
|
|
|
|
|
|
|
|
if (!ctx.cur) {
|
|
|
|
weston_log("Timeline error in fmemopen, closing.\n");
|
|
|
|
weston_timeline_close();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(ctx.cur, "{ \"T\":[%" PRId64 ", %ld], \"N\":\"%s\"",
|
|
|
|
(int64_t)ts.tv_sec, ts.tv_nsec, name);
|
|
|
|
|
|
|
|
va_start(argp, name);
|
|
|
|
while (1) {
|
|
|
|
otype = va_arg(argp, enum timeline_type);
|
|
|
|
if (otype == TLT_END)
|
|
|
|
break;
|
|
|
|
|
|
|
|
obj = va_arg(argp, void *);
|
|
|
|
if (type_dispatch[otype]) {
|
|
|
|
fprintf(ctx.cur, ", ");
|
|
|
|
type_dispatch[otype](&ctx, obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
va_end(argp);
|
|
|
|
|
|
|
|
fprintf(ctx.cur, " }\n");
|
|
|
|
fflush(ctx.cur);
|
|
|
|
if (ferror(ctx.cur)) {
|
|
|
|
weston_log("Timeline error in constructing entry, closing.\n");
|
|
|
|
weston_timeline_close();
|
|
|
|
} else {
|
|
|
|
fprintf(ctx.out, "%s", buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(ctx.cur);
|
|
|
|
}
|