2 * Copyright (C) 2019 Simon Marchi <simon.marchi@efficios.com>
4 * SPDX-License-Identifier: LGPL-2.1-only
9 #include <common/error.h>
10 #include <common/macros.h>
11 #include <common/snapshot.h>
12 #include <common/payload.h>
13 #include <common/payload-view.h>
14 #include <lttng/action/action-internal.h>
15 #include <lttng/action/snapshot-session-internal.h>
16 #include <lttng/action/snapshot-session.h>
17 #include <lttng/snapshot.h>
18 #include <lttng/snapshot-internal.h>
21 #define IS_SNAPSHOT_SESSION_ACTION(action) \
22 (lttng_action_get_type(action) == LTTNG_ACTION_TYPE_SNAPSHOT_SESSION)
24 struct lttng_action_snapshot_session
{
25 struct lttng_action parent
;
31 * When non-NULL, use this custom output when taking the snapshot,
32 * rather than the session's registered snapshot output.
36 struct lttng_snapshot_output
*output
;
39 struct lttng_action_snapshot_session_comm
{
40 /* All string lengths include the trailing \0. */
41 uint32_t session_name_len
;
42 uint32_t snapshot_output_len
;
45 * Variable data (all strings are null-terminated):
47 * - session name string
48 * - snapshot output object
54 static struct lttng_action_snapshot_session
*
55 action_snapshot_session_from_action(struct lttng_action
*action
)
60 action
, struct lttng_action_snapshot_session
, parent
);
63 static const struct lttng_action_snapshot_session
*
64 action_snapshot_session_from_action_const(const struct lttng_action
*action
)
69 action
, struct lttng_action_snapshot_session
, parent
);
72 static bool lttng_action_snapshot_session_validate(struct lttng_action
*action
)
75 struct lttng_action_snapshot_session
*action_snapshot_session
;
81 action_snapshot_session
= action_snapshot_session_from_action(action
);
83 /* A non-empty session name is mandatory. */
84 if (!action_snapshot_session
->session_name
||
85 strlen(action_snapshot_session
->session_name
) == 0) {
89 if (action_snapshot_session
->output
&&
90 !lttng_snapshot_output_validate(action_snapshot_session
->output
)) {
99 static bool lttng_action_snapshot_session_is_equal(
100 const struct lttng_action
*_a
, const struct lttng_action
*_b
)
102 bool is_equal
= false;
103 const struct lttng_action_snapshot_session
*a
, *b
;
105 a
= action_snapshot_session_from_action_const(_a
);
106 b
= action_snapshot_session_from_action_const(_b
);
108 /* Action is not valid if this is not true. */
109 assert(a
->session_name
);
110 assert(b
->session_name
);
111 if (strcmp(a
->session_name
, b
->session_name
)) {
115 if (a
->output
&& b
->output
&&
116 !lttng_snapshot_output_is_equal(a
->output
, b
->output
)) {
118 } else if (!!a
->output
!= !!b
->output
) {
127 static size_t serialize_strlen(const char *str
)
129 return str
? strlen(str
) + 1 : 0;
132 static int lttng_action_snapshot_session_serialize(
133 struct lttng_action
*action
, struct lttng_payload
*payload
)
135 struct lttng_action_snapshot_session
*action_snapshot_session
;
136 struct lttng_action_snapshot_session_comm comm
= {};
138 size_t size_before_comm
;
143 size_before_comm
= payload
->buffer
.size
;
145 action_snapshot_session
= action_snapshot_session_from_action(action
);
146 comm
.session_name_len
=
147 serialize_strlen(action_snapshot_session
->session_name
);
150 ret
= lttng_dynamic_buffer_append(
151 &payload
->buffer
, &comm
, sizeof(comm
));
156 assert(action_snapshot_session
->session_name
);
157 DBG("Serializing snapshot session action: session-name: %s",
158 action_snapshot_session
->session_name
);
160 /* Add session name. */
161 ret
= lttng_dynamic_buffer_append(&payload
->buffer
,
162 action_snapshot_session
->session_name
,
163 comm
.session_name_len
);
168 /* Serialize the snapshot output object, if any. */
169 if (action_snapshot_session
->output
) {
170 const size_t size_before_output
= payload
->buffer
.size
;
171 struct lttng_action_snapshot_session_comm
*comm_in_payload
;
173 ret
= lttng_snapshot_output_serialize(
174 action_snapshot_session
->output
,
180 /* Adjust action length in header. */
181 comm_in_payload
= (typeof(comm_in_payload
))(
182 payload
->buffer
.data
+ size_before_comm
);
183 comm_in_payload
->snapshot_output_len
=
184 payload
->buffer
.size
- size_before_output
;
191 static void lttng_action_snapshot_session_destroy(struct lttng_action
*action
)
193 struct lttng_action_snapshot_session
*action_snapshot_session
;
199 action_snapshot_session
= action_snapshot_session_from_action(action
);
201 free(action_snapshot_session
->session_name
);
202 lttng_snapshot_output_destroy(action_snapshot_session
->output
);
203 free(action_snapshot_session
);
209 ssize_t
lttng_action_snapshot_session_create_from_payload(
210 struct lttng_payload_view
*view
,
211 struct lttng_action
**p_action
)
213 ssize_t consumed_len
;
214 const struct lttng_action_snapshot_session_comm
*comm
;
215 const char *variable_data
;
216 struct lttng_action
*action
;
217 enum lttng_action_status status
;
218 struct lttng_snapshot_output
*snapshot_output
= NULL
;
220 action
= lttng_action_snapshot_session_create();
225 comm
= (typeof(comm
)) view
->buffer
.data
;
226 variable_data
= (const char *) &comm
->data
;
228 consumed_len
= sizeof(struct lttng_action_snapshot_session_comm
);
230 if (!lttng_buffer_view_contains_string(
231 &view
->buffer
, variable_data
, comm
->session_name_len
)) {
235 status
= lttng_action_snapshot_session_set_session_name(
236 action
, variable_data
);
237 if (status
!= LTTNG_ACTION_STATUS_OK
) {
241 variable_data
+= comm
->session_name_len
;
242 consumed_len
+= comm
->session_name_len
;
244 /* If there is a snapshot output object, deserialize it. */
245 if (comm
->snapshot_output_len
> 0) {
246 ssize_t snapshot_output_consumed_len
;
247 enum lttng_action_status action_status
;
248 struct lttng_payload_view snapshot_output_buffer_view
=
249 lttng_payload_view_from_view(view
, consumed_len
,
250 comm
->snapshot_output_len
);
252 if (!snapshot_output_buffer_view
.buffer
.data
) {
253 ERR("Failed to create buffer view for snapshot output.");
257 snapshot_output_consumed_len
=
258 lttng_snapshot_output_create_from_payload(
259 &snapshot_output_buffer_view
,
261 if (snapshot_output_consumed_len
!= comm
->snapshot_output_len
) {
262 ERR("Failed to deserialize snapshot output object: "
263 "consumed-len: %zd, expected-len: %" PRIu32
,
264 snapshot_output_consumed_len
,
265 comm
->snapshot_output_len
);
269 action_status
= lttng_action_snapshot_session_set_output(
270 action
, snapshot_output
);
271 if (action_status
!= LTTNG_ACTION_STATUS_OK
) {
275 /* Ownership has been transferred to the action. */
276 snapshot_output
= NULL
;
279 variable_data
+= comm
->snapshot_output_len
;
280 consumed_len
+= comm
->snapshot_output_len
;
290 lttng_action_snapshot_session_destroy(action
);
291 lttng_snapshot_output_destroy(snapshot_output
);
296 struct lttng_action
*lttng_action_snapshot_session_create(void)
298 struct lttng_action
*action
;
300 action
= zmalloc(sizeof(struct lttng_action_snapshot_session
));
305 lttng_action_init(action
, LTTNG_ACTION_TYPE_SNAPSHOT_SESSION
,
306 lttng_action_snapshot_session_validate
,
307 lttng_action_snapshot_session_serialize
,
308 lttng_action_snapshot_session_is_equal
,
309 lttng_action_snapshot_session_destroy
);
315 enum lttng_action_status
lttng_action_snapshot_session_set_session_name(
316 struct lttng_action
*action
, const char *session_name
)
318 struct lttng_action_snapshot_session
*action_snapshot_session
;
319 enum lttng_action_status status
;
321 if (!action
|| !IS_SNAPSHOT_SESSION_ACTION(action
) || !session_name
||
322 strlen(session_name
) == 0) {
323 status
= LTTNG_ACTION_STATUS_INVALID
;
327 action_snapshot_session
= action_snapshot_session_from_action(action
);
329 free(action_snapshot_session
->session_name
);
331 action_snapshot_session
->session_name
= strdup(session_name
);
332 if (!action_snapshot_session
->session_name
) {
333 status
= LTTNG_ACTION_STATUS_ERROR
;
337 status
= LTTNG_ACTION_STATUS_OK
;
342 enum lttng_action_status
lttng_action_snapshot_session_get_session_name(
343 const struct lttng_action
*action
, const char **session_name
)
345 const struct lttng_action_snapshot_session
*action_snapshot_session
;
346 enum lttng_action_status status
;
348 if (!action
|| !IS_SNAPSHOT_SESSION_ACTION(action
) || !session_name
) {
349 status
= LTTNG_ACTION_STATUS_INVALID
;
353 action_snapshot_session
= action_snapshot_session_from_action_const(action
);
355 if (action_snapshot_session
->session_name
) {
356 *session_name
= action_snapshot_session
->session_name
;
357 status
= LTTNG_ACTION_STATUS_OK
;
359 status
= LTTNG_ACTION_STATUS_UNSET
;
367 enum lttng_action_status
lttng_action_snapshot_session_set_output(
368 struct lttng_action
*action
,
369 struct lttng_snapshot_output
*output
)
371 struct lttng_action_snapshot_session
*action_snapshot_session
;
372 enum lttng_action_status status
;
374 if (!action
|| !IS_SNAPSHOT_SESSION_ACTION(action
) || !output
) {
375 status
= LTTNG_ACTION_STATUS_INVALID
;
379 action_snapshot_session
= action_snapshot_session_from_action(action
);
381 lttng_snapshot_output_destroy(action_snapshot_session
->output
);
382 action_snapshot_session
->output
= output
;
384 status
= LTTNG_ACTION_STATUS_OK
;
390 enum lttng_action_status
lttng_action_snapshot_session_get_output(
391 const struct lttng_action
*action
,
392 const struct lttng_snapshot_output
**output
)
394 const struct lttng_action_snapshot_session
*action_snapshot_session
;
395 enum lttng_action_status status
;
397 if (!action
|| !IS_SNAPSHOT_SESSION_ACTION(action
)|| !output
) {
398 status
= LTTNG_ACTION_STATUS_INVALID
;
402 action_snapshot_session
= action_snapshot_session_from_action_const(action
);
404 if (action_snapshot_session
->output
) {
405 *output
= action_snapshot_session
->output
;
406 status
= LTTNG_ACTION_STATUS_OK
;
408 status
= LTTNG_ACTION_STATUS_UNSET
;