Fix: unchecked buffer size for communication header
[lttng-tools.git] / src / common / actions / action.c
CommitLineData
a58c490f 1/*
ab5be9fa 2 * Copyright (C) 2017 Jérémie Galarneau <jeremie.galarneau@efficios.com>
a58c490f 3 *
ab5be9fa 4 * SPDX-License-Identifier: LGPL-2.1-only
a58c490f 5 *
a58c490f
JG
6 */
7
14ec7e87
JR
8#include <assert.h>
9#include <common/error.h>
a58c490f 10#include <lttng/action/action-internal.h>
0c51e8f3 11#include <lttng/action/group-internal.h>
a58c490f 12#include <lttng/action/notify-internal.h>
bfb2ec6a 13#include <lttng/action/rotate-session-internal.h>
757c48a2 14#include <lttng/action/snapshot-session-internal.h>
58397d0d 15#include <lttng/action/start-session-internal.h>
931bdbaa 16#include <lttng/action/stop-session-internal.h>
a58c490f 17
10615eee
JR
18LTTNG_HIDDEN
19const char *lttng_action_type_string(enum lttng_action_type action_type)
2666d352
SM
20{
21 switch (action_type) {
22 case LTTNG_ACTION_TYPE_UNKNOWN:
23 return "UNKNOWN";
0c51e8f3
SM
24 case LTTNG_ACTION_TYPE_GROUP:
25 return "GROUP";
2666d352
SM
26 case LTTNG_ACTION_TYPE_NOTIFY:
27 return "NOTIFY";
bfb2ec6a
SM
28 case LTTNG_ACTION_TYPE_ROTATE_SESSION:
29 return "ROTATE_SESSION";
757c48a2
SM
30 case LTTNG_ACTION_TYPE_SNAPSHOT_SESSION:
31 return "SNAPSHOT_SESSION";
58397d0d
SM
32 case LTTNG_ACTION_TYPE_START_SESSION:
33 return "START_SESSION";
931bdbaa
SM
34 case LTTNG_ACTION_TYPE_STOP_SESSION:
35 return "STOP_SESSION";
2666d352
SM
36 default:
37 return "???";
38 }
39}
40
17182cfd 41enum lttng_action_type lttng_action_get_type(const struct lttng_action *action)
a58c490f
JG
42{
43 return action ? action->type : LTTNG_ACTION_TYPE_UNKNOWN;
44}
45
6acb3f46
SM
46LTTNG_HIDDEN
47void lttng_action_init(
48 struct lttng_action *action,
49 enum lttng_action_type type,
50 action_validate_cb validate,
51 action_serialize_cb serialize,
3dd04a6a 52 action_equal_cb equal,
6acb3f46
SM
53 action_destroy_cb destroy)
54{
c852ce4e 55 urcu_ref_init(&action->ref);
6acb3f46
SM
56 action->type = type;
57 action->validate = validate;
58 action->serialize = serialize;
3dd04a6a 59 action->equal = equal;
6acb3f46
SM
60 action->destroy = destroy;
61}
62
c852ce4e
JG
63static
64void action_destroy_ref(struct urcu_ref *ref)
65{
66 struct lttng_action *action =
67 container_of(ref, struct lttng_action, ref);
68
69 action->destroy(action);
70}
71
72LTTNG_HIDDEN
73void lttng_action_get(struct lttng_action *action)
74{
75 urcu_ref_get(&action->ref);
76}
77
78LTTNG_HIDDEN
79void lttng_action_put(struct lttng_action *action)
a58c490f
JG
80{
81 if (!action) {
82 return;
83 }
84
85 assert(action->destroy);
c852ce4e
JG
86 urcu_ref_put(&action->ref, action_destroy_ref);
87}
88
89void lttng_action_destroy(struct lttng_action *action)
90{
91 lttng_action_put(action);
a58c490f
JG
92}
93
94LTTNG_HIDDEN
95bool lttng_action_validate(struct lttng_action *action)
96{
97 bool valid;
98
99 if (!action) {
100 valid = false;
101 goto end;
102 }
103
104 if (!action->validate) {
105 /* Sub-class guarantees that it can never be invalid. */
106 valid = true;
107 goto end;
108 }
109
110 valid = action->validate(action);
111end:
112 return valid;
113}
114
115LTTNG_HIDDEN
3647288f 116int lttng_action_serialize(struct lttng_action *action,
c0a66c84 117 struct lttng_payload *payload)
a58c490f 118{
3647288f
JG
119 int ret;
120 struct lttng_action_comm action_comm = {
121 .action_type = (int8_t) action->type,
122 };
123
c0a66c84 124 ret = lttng_dynamic_buffer_append(&payload->buffer, &action_comm,
3647288f
JG
125 sizeof(action_comm));
126 if (ret) {
a58c490f
JG
127 goto end;
128 }
129
c0a66c84 130 ret = action->serialize(action, payload);
3647288f 131 if (ret) {
a58c490f
JG
132 goto end;
133 }
a58c490f
JG
134end:
135 return ret;
136}
137
138LTTNG_HIDDEN
c0a66c84 139ssize_t lttng_action_create_from_payload(struct lttng_payload_view *view,
869a3c2d 140 struct lttng_action **action)
a58c490f 141{
869a3c2d 142 ssize_t consumed_len, specific_action_consumed_len;
c0a66c84 143 action_create_from_payload_cb create_from_payload_cb;
3e6e0df2
JG
144 const struct lttng_action_comm *action_comm;
145 const struct lttng_payload_view action_comm_view =
146 lttng_payload_view_from_view(
147 view, 0, sizeof(*action_comm));
a58c490f 148
869a3c2d
SM
149 if (!view || !action) {
150 consumed_len = -1;
a58c490f
JG
151 goto end;
152 }
153
3e6e0df2
JG
154 if (!lttng_payload_view_is_valid(&action_comm_view)) {
155 /* Payload not large enough to contain the header. */
156 consumed_len = -1;
157 goto end;
158 }
159
160 action_comm = (const struct lttng_action_comm *) action_comm_view.buffer.data;
869a3c2d 161
c0a66c84 162 DBG("Create action from payload: action-type=%s",
2666d352
SM
163 lttng_action_type_string(action_comm->action_type));
164
a58c490f
JG
165 switch (action_comm->action_type) {
166 case LTTNG_ACTION_TYPE_NOTIFY:
c0a66c84 167 create_from_payload_cb = lttng_action_notify_create_from_payload;
a58c490f 168 break;
bfb2ec6a 169 case LTTNG_ACTION_TYPE_ROTATE_SESSION:
c0a66c84
JG
170 create_from_payload_cb =
171 lttng_action_rotate_session_create_from_payload;
bfb2ec6a 172 break;
757c48a2
SM
173 case LTTNG_ACTION_TYPE_SNAPSHOT_SESSION:
174 create_from_payload_cb =
175 lttng_action_snapshot_session_create_from_payload;
176 break;
58397d0d 177 case LTTNG_ACTION_TYPE_START_SESSION:
c0a66c84
JG
178 create_from_payload_cb =
179 lttng_action_start_session_create_from_payload;
58397d0d 180 break;
931bdbaa 181 case LTTNG_ACTION_TYPE_STOP_SESSION:
c0a66c84
JG
182 create_from_payload_cb =
183 lttng_action_stop_session_create_from_payload;
931bdbaa 184 break;
0c51e8f3
SM
185 case LTTNG_ACTION_TYPE_GROUP:
186 create_from_payload_cb = lttng_action_group_create_from_payload;
187 break;
a58c490f 188 default:
c0a66c84 189 ERR("Failed to create action from payload, unhandled action type: action-type=%u (%s)",
2666d352
SM
190 action_comm->action_type,
191 lttng_action_type_string(
192 action_comm->action_type));
869a3c2d 193 consumed_len = -1;
a58c490f
JG
194 goto end;
195 }
196
c0a66c84
JG
197 {
198 /* Create buffer view for the action-type-specific data. */
199 struct lttng_payload_view specific_action_view =
200 lttng_payload_view_from_view(view,
201 sizeof(struct lttng_action_comm),
202 -1);
869a3c2d 203
c0a66c84
JG
204 specific_action_consumed_len = create_from_payload_cb(
205 &specific_action_view, action);
206 }
869a3c2d
SM
207 if (specific_action_consumed_len < 0) {
208 ERR("Failed to create specific action from buffer.");
209 consumed_len = -1;
a58c490f
JG
210 goto end;
211 }
869a3c2d
SM
212
213 assert(*action);
214
215 consumed_len = sizeof(struct lttng_action_comm) +
216 specific_action_consumed_len;
217
a58c490f 218end:
869a3c2d 219 return consumed_len;
a58c490f 220}
3dd04a6a
JR
221
222LTTNG_HIDDEN
223bool lttng_action_is_equal(const struct lttng_action *a,
224 const struct lttng_action *b)
225{
226 bool is_equal = false;
227
228 if (!a || !b) {
229 goto end;
230 }
231
232 if (a->type != b->type) {
233 goto end;
234 }
235
236 if (a == b) {
237 is_equal = true;
238 goto end;
239 }
240
241 assert(a->equal);
242 is_equal = a->equal(a, b);
243end:
244 return is_equal;
245}
This page took 0.043914 seconds and 4 git commands to generate.