1 From acbce75de9587917cfa659ebc0e3404b6b1d4c29 Mon Sep 17 00:00:00 2001
2 From: George Kiagiadakis <george.kiagiadakis@collabora.com>
3 Date: Thu, 23 May 2019 18:59:05 +0300
4 Subject: [PATCH] extensions: implement new session manager extension
6 This extension, implemented in module-session-manager, implements
7 a set of objects that are useful for session managers.
9 Upstream-Status: Pending
11 src/extensions/meson.build | 9 +
12 src/extensions/session-manager.h | 34 +
13 .../session-manager/impl-interfaces.h | 329 +++
14 src/extensions/session-manager/interfaces.h | 465 ++++
15 src/extensions/session-manager/introspect.h | 131 +
16 src/extensions/session-manager/keys.h | 40 +
17 src/modules/meson.build | 17 +
18 src/modules/module-session-manager.c | 56 +
19 .../module-session-manager/client-endpoint.c | 270 +++
20 .../module-session-manager/client-endpoint.h | 60 +
21 .../module-session-manager/client-session.c | 270 +++
22 .../module-session-manager/client-session.h | 62 +
23 .../module-session-manager/endpoint-link.c | 359 +++
24 .../module-session-manager/endpoint-link.h | 64 +
25 .../module-session-manager/endpoint-stream.c | 329 +++
26 .../module-session-manager/endpoint-stream.h | 64 +
27 src/modules/module-session-manager/endpoint.c | 343 +++
28 src/modules/module-session-manager/endpoint.h | 61 +
29 .../module-session-manager/protocol-native.c | 2125 +++++++++++++++++
30 src/modules/module-session-manager/session.c | 341 +++
31 src/modules/module-session-manager/session.h | 61 +
32 src/pipewire/pipewire.c | 6 +
33 src/pipewire/type.h | 7 +-
34 23 files changed, 5502 insertions(+), 1 deletion(-)
35 create mode 100644 src/extensions/session-manager.h
36 create mode 100644 src/extensions/session-manager/impl-interfaces.h
37 create mode 100644 src/extensions/session-manager/interfaces.h
38 create mode 100644 src/extensions/session-manager/introspect.h
39 create mode 100644 src/extensions/session-manager/keys.h
40 create mode 100644 src/modules/module-session-manager.c
41 create mode 100644 src/modules/module-session-manager/client-endpoint.c
42 create mode 100644 src/modules/module-session-manager/client-endpoint.h
43 create mode 100644 src/modules/module-session-manager/client-session.c
44 create mode 100644 src/modules/module-session-manager/client-session.h
45 create mode 100644 src/modules/module-session-manager/endpoint-link.c
46 create mode 100644 src/modules/module-session-manager/endpoint-link.h
47 create mode 100644 src/modules/module-session-manager/endpoint-stream.c
48 create mode 100644 src/modules/module-session-manager/endpoint-stream.h
49 create mode 100644 src/modules/module-session-manager/endpoint.c
50 create mode 100644 src/modules/module-session-manager/endpoint.h
51 create mode 100644 src/modules/module-session-manager/protocol-native.c
52 create mode 100644 src/modules/module-session-manager/session.c
53 create mode 100644 src/modules/module-session-manager/session.h
55 diff --git a/src/extensions/meson.build b/src/extensions/meson.build
56 index a7f5d3cb..95377faa 100644
57 --- a/src/extensions/meson.build
58 +++ b/src/extensions/meson.build
60 +pipewire_ext_sm_headers = [
61 + 'session-manager/impl-interfaces.h',
62 + 'session-manager/interfaces.h',
63 + 'session-manager/introspect.h',
64 + 'session-manager/keys.h',
67 pipewire_ext_headers = [
70 + 'session-manager.h',
73 +install_headers(pipewire_ext_sm_headers, subdir : 'pipewire/extensions/session-manager')
74 install_headers(pipewire_ext_headers, subdir : 'pipewire/extensions')
75 diff --git a/src/extensions/session-manager.h b/src/extensions/session-manager.h
77 index 00000000..95e759b0
79 +++ b/src/extensions/session-manager.h
83 + * Copyright © 2019 Collabora Ltd.
84 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
86 + * Permission is hereby granted, free of charge, to any person obtaining a
87 + * copy of this software and associated documentation files (the "Software"),
88 + * to deal in the Software without restriction, including without limitation
89 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
90 + * and/or sell copies of the Software, and to permit persons to whom the
91 + * Software is furnished to do so, subject to the following conditions:
93 + * The above copyright notice and this permission notice (including the next
94 + * paragraph) shall be included in all copies or substantial portions of the
97 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
98 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
99 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
100 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
101 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
102 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
103 + * DEALINGS IN THE SOFTWARE.
106 +#ifndef PIPEWIRE_EXT_SESSION_MANAGER_H
107 +#define PIPEWIRE_EXT_SESSION_MANAGER_H
109 +#include "session-manager/introspect.h"
110 +#include "session-manager/interfaces.h"
111 +#include "session-manager/impl-interfaces.h"
112 +#include "session-manager/keys.h"
114 +#endif /* PIPEWIRE_EXT_SESSION_MANAGER_H */
115 diff --git a/src/extensions/session-manager/impl-interfaces.h b/src/extensions/session-manager/impl-interfaces.h
117 index 00000000..66daa0b9
119 +++ b/src/extensions/session-manager/impl-interfaces.h
123 + * Copyright © 2019 Collabora Ltd.
124 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
126 + * Permission is hereby granted, free of charge, to any person obtaining a
127 + * copy of this software and associated documentation files (the "Software"),
128 + * to deal in the Software without restriction, including without limitation
129 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
130 + * and/or sell copies of the Software, and to permit persons to whom the
131 + * Software is furnished to do so, subject to the following conditions:
133 + * The above copyright notice and this permission notice (including the next
134 + * paragraph) shall be included in all copies or substantial portions of the
137 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
138 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
139 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
140 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
141 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
142 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
143 + * DEALINGS IN THE SOFTWARE.
146 +#ifndef PIPEWIRE_EXT_SESSION_MANAGER_IMPL_INTERFACES_H
147 +#define PIPEWIRE_EXT_SESSION_MANAGER_IMPL_INTERFACES_H
149 +#include <spa/utils/defs.h>
150 +#include <spa/utils/hook.h>
153 +#include "introspect.h"
159 +#define PW_VERSION_CLIENT_ENDPOINT_PROXY 0
160 +struct pw_client_endpoint_proxy { struct spa_interface iface; };
162 +#define PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_ID 0
163 +#define PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_SESSION_ID 1
164 +#define PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_PARAM 2
165 +#define PW_CLIENT_ENDPOINT_PROXY_EVENT_STREAM_SET_PARAM 3
166 +#define PW_CLIENT_ENDPOINT_PROXY_EVENT_NUM 4
168 +struct pw_client_endpoint_proxy_events {
169 +#define PW_VERSION_CLIENT_ENDPOINT_PROXY_EVENTS 0
170 + uint32_t version; /**< version of this structure */
173 + * Sets the id of the \a endpoint.
175 + * On endpoint implementations, this is called by the server to notify
176 + * the implementation of the assigned global id of the endpoint. The
177 + * implementation is obliged to set this id in the
178 + * #struct pw_endpoint_info \a id field. The implementation should also
179 + * not emit the info() event before this method is called.
181 + * \param endpoint a #pw_endpoint
182 + * \param id the global id assigned to this endpoint
184 + * \return 0 on success
185 + * -EINVAL when the id has already been set
186 + * -ENOTSUP on the server-side endpoint implementation
188 + int (*set_id) (void *endpoint, uint32_t id);
191 + * Sets the session id of the \a endpoint.
193 + * On endpoints that are not session masters, this method notifies
194 + * the implementation that it has been associated with a session.
195 + * The implementation is obliged to set this id in the
196 + * #struct pw_endpoint_info \a session_id field.
198 + * \param endpoint a #pw_endpoint
199 + * \param id the session id associated with this endpoint
201 + * \return 0 on success
202 + * -EINVAL when the session id has already been set
203 + * -ENOTSUP when the endpoint is a session master
205 + int (*set_session_id) (void *endpoint, uint32_t session_id);
208 + * Set the configurable parameter in \a endpoint.
210 + * Usually, \a param will be obtained from enum_params and then
211 + * modified but it is also possible to set another spa_pod
212 + * as long as its keys and types match a supported object.
214 + * Objects with property keys that are not known are ignored.
216 + * This function must be called from the main thread.
218 + * \param endpoint a #struct pw_endpoint
219 + * \param id the parameter id to configure
220 + * \param flags additional flags
221 + * \param param the parameter to configure
223 + * \return 0 on success
224 + * -EINVAL when \a endpoint is NULL
225 + * -ENOTSUP when there are no parameters implemented on \a endpoint
226 + * -ENOENT the parameter is unknown
228 + int (*set_param) (void *endpoint,
229 + uint32_t id, uint32_t flags,
230 + const struct spa_pod *param);
233 + * Set a parameter on \a stream_id of \a endpoint.
235 + * When \a param is NULL, the parameter will be unset.
237 + * This function must be called from the main thread.
239 + * \param endpoint a #struct pw_endpoint
240 + * \param stream_id the stream to configure
241 + * \param id the parameter id to set
242 + * \param flags optional flags
243 + * \param param a #struct spa_pod with the parameter to set
244 + * \return 0 on success
245 + * 1 on success, the value of \a param might have been
246 + * changed depending on \a flags and the final value can
247 + * be found by doing stream_enum_params.
248 + * -EINVAL when \a endpoint is NULL or invalid arguments are given
249 + * -ESRCH when the type or size of a property is not correct.
250 + * -ENOENT when the param id is not found
252 + int (*stream_set_param) (void *endpoint, uint32_t stream_id,
253 + uint32_t id, uint32_t flags,
254 + const struct spa_pod *param);
257 +#define PW_CLIENT_ENDPOINT_PROXY_METHOD_ADD_LISTENER 0
258 +#define PW_CLIENT_ENDPOINT_PROXY_METHOD_UPDATE 1
259 +#define PW_CLIENT_ENDPOINT_PROXY_METHOD_STREAM_UPDATE 2
260 +#define PW_CLIENT_ENDPOINT_PROXY_METHOD_NUM 3
262 +struct pw_client_endpoint_proxy_methods {
263 +#define PW_VERSION_CLIENT_ENDPOINT_PROXY_METHODS 0
264 + uint32_t version; /**< version of this structure */
266 + int (*add_listener) (void *object,
267 + struct spa_hook *listener,
268 + const struct pw_client_endpoint_proxy_events *events,
271 + /** Update endpoint information */
272 + int (*update) (void *object,
273 +#define PW_CLIENT_ENDPOINT_UPDATE_PARAMS (1 << 0)
274 +#define PW_CLIENT_ENDPOINT_UPDATE_INFO (1 << 1)
275 + uint32_t change_mask,
277 + const struct spa_pod **params,
278 + const struct pw_endpoint_info *info);
280 + /** Update stream information */
281 + int (*stream_update) (void *object,
282 + uint32_t stream_id,
283 +#define PW_CLIENT_ENDPOINT_STREAM_UPDATE_PARAMS (1 << 0)
284 +#define PW_CLIENT_ENDPOINT_STREAM_UPDATE_INFO (1 << 1)
285 +#define PW_CLIENT_ENDPOINT_STREAM_UPDATE_DESTROYED (1 << 2)
286 + uint32_t change_mask,
288 + const struct spa_pod **params,
289 + const struct pw_endpoint_stream_info *info);
292 +#define pw_client_endpoint_proxy_method(o,method,version,...) \
294 + int _res = -ENOTSUP; \
295 + struct pw_client_endpoint_proxy *_p = o; \
296 + spa_interface_call_res(&_p->iface, \
297 + struct pw_client_endpoint_proxy_methods, _res, \
298 + method, version, ##__VA_ARGS__); \
302 +#define pw_client_endpoint_proxy_add_listener(o,...) pw_client_endpoint_proxy_method(o,add_listener,0,__VA_ARGS__)
303 +#define pw_client_endpoint_proxy_update(o,...) pw_client_endpoint_proxy_method(o,update,0,__VA_ARGS__)
304 +#define pw_client_endpoint_proxy_stream_update(o,...) pw_client_endpoint_proxy_method(o,stream_update,0,__VA_ARGS__)
307 +#define PW_VERSION_CLIENT_SESSION_PROXY 0
308 +struct pw_client_session_proxy { struct spa_interface iface; };
310 +#define PW_CLIENT_SESSION_PROXY_EVENT_SET_ID 0
311 +#define PW_CLIENT_SESSION_PROXY_EVENT_SET_PARAM 1
312 +#define PW_CLIENT_SESSION_PROXY_EVENT_LINK_SET_PARAM 2
313 +#define PW_CLIENT_SESSION_PROXY_EVENT_CREATE_LINK 3
314 +#define PW_CLIENT_SESSION_PROXY_EVENT_DESTROY_LINK 4
315 +#define PW_CLIENT_SESSION_PROXY_EVENT_LINK_REQUEST_STATE 5
316 +#define PW_CLIENT_SESSION_PROXY_EVENT_NUM 6
318 +struct pw_client_session_proxy_events {
319 +#define PW_VERSION_CLIENT_SESSION_PROXY_EVENTS 0
320 + uint32_t version; /**< version of this structure */
323 + * Sets the id of the \a session.
325 + * On session implementations, this is called by the server to notify
326 + * the implementation of the assigned global id of the session. The
327 + * implementation is obliged to set this id in the
328 + * #struct pw_session_info \a id field. The implementation should also
329 + * not emit the info() event before this method is called.
331 + * \param session a #pw_session
332 + * \param id the global id assigned to this session
334 + * \return 0 on success
335 + * -EINVAL when the id has already been set
336 + * -ENOTSUP on the server-side session implementation
338 + int (*set_id) (void *session, uint32_t id);
341 + * Set the configurable parameter in \a session.
343 + * Usually, \a param will be obtained from enum_params and then
344 + * modified but it is also possible to set another spa_pod
345 + * as long as its keys and types match a supported object.
347 + * Objects with property keys that are not known are ignored.
349 + * This function must be called from the main thread.
351 + * \param session a #struct pw_session
352 + * \param id the parameter id to configure
353 + * \param flags additional flags
354 + * \param param the parameter to configure
356 + * \return 0 on success
357 + * -EINVAL when \a session is NULL
358 + * -ENOTSUP when there are no parameters implemented on \a session
359 + * -ENOENT the parameter is unknown
361 + int (*set_param) (void *session,
362 + uint32_t id, uint32_t flags,
363 + const struct spa_pod *param);
366 + * Set a parameter on \a link_id of \a session.
368 + * When \a param is NULL, the parameter will be unset.
370 + * This function must be called from the main thread.
372 + * \param session a #struct pw_session
373 + * \param link_id the link to configure
374 + * \param id the parameter id to set
375 + * \param flags optional flags
376 + * \param param a #struct spa_pod with the parameter to set
377 + * \return 0 on success
378 + * 1 on success, the value of \a param might have been
379 + * changed depending on \a flags and the final value can
380 + * be found by doing link_enum_params.
381 + * -EINVAL when \a session is NULL or invalid arguments are given
382 + * -ESRCH when the type or size of a property is not correct.
383 + * -ENOENT when the param id is not found
385 + int (*link_set_param) (void *session, uint32_t link_id,
386 + uint32_t id, uint32_t flags,
387 + const struct spa_pod *param);
389 + int (*create_link) (void *session, const struct spa_dict *props);
391 + int (*destroy_link) (void *session, uint32_t link_id);
393 + int (*link_request_state) (void *session, uint32_t link_id, uint32_t state);
396 +#define PW_CLIENT_SESSION_PROXY_METHOD_ADD_LISTENER 0
397 +#define PW_CLIENT_SESSION_PROXY_METHOD_UPDATE 1
398 +#define PW_CLIENT_SESSION_PROXY_METHOD_LINK_UPDATE 2
399 +#define PW_CLIENT_SESSION_PROXY_METHOD_NUM 3
401 +struct pw_client_session_proxy_methods {
402 +#define PW_VERSION_CLIENT_SESSION_PROXY_METHODS 0
403 + uint32_t version; /**< version of this structure */
405 + int (*add_listener) (void *object,
406 + struct spa_hook *listener,
407 + const struct pw_client_session_proxy_events *events,
410 + /** Update session information */
411 + int (*update) (void *object,
412 +#define PW_CLIENT_SESSION_UPDATE_PARAMS (1 << 0)
413 +#define PW_CLIENT_SESSION_UPDATE_INFO (1 << 1)
414 + uint32_t change_mask,
416 + const struct spa_pod **params,
417 + const struct pw_session_info *info);
419 + /** Update link information */
420 + int (*link_update) (void *object,
422 +#define PW_CLIENT_SESSION_LINK_UPDATE_PARAMS (1 << 0)
423 +#define PW_CLIENT_SESSION_LINK_UPDATE_INFO (1 << 1)
424 +#define PW_CLIENT_SESSION_LINK_UPDATE_DESTROYED (1 << 2)
425 + uint32_t change_mask,
427 + const struct spa_pod **params,
428 + const struct pw_endpoint_link_info *info);
431 +#define pw_client_session_proxy_method(o,method,version,...) \
433 + int _res = -ENOTSUP; \
434 + struct pw_client_session_proxy *_p = o; \
435 + spa_interface_call_res(&_p->iface, \
436 + struct pw_client_session_proxy_methods, _res, \
437 + method, version, ##__VA_ARGS__); \
441 +#define pw_client_session_proxy_add_listener(o,...) pw_client_session_proxy_method(o,add_listener,0,__VA_ARGS__)
442 +#define pw_client_session_proxy_update(o,...) pw_client_session_proxy_method(o,update,0,__VA_ARGS__)
443 +#define pw_client_session_proxy_link_update(o,...) pw_client_session_proxy_method(o,link_update,0,__VA_ARGS__)
449 +#endif /* PIPEWIRE_EXT_SESSION_MANAGER_IMPL_INTERFACES_H */
450 diff --git a/src/extensions/session-manager/interfaces.h b/src/extensions/session-manager/interfaces.h
452 index 00000000..0651e8bf
454 +++ b/src/extensions/session-manager/interfaces.h
458 + * Copyright © 2019 Collabora Ltd.
459 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
461 + * Permission is hereby granted, free of charge, to any person obtaining a
462 + * copy of this software and associated documentation files (the "Software"),
463 + * to deal in the Software without restriction, including without limitation
464 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
465 + * and/or sell copies of the Software, and to permit persons to whom the
466 + * Software is furnished to do so, subject to the following conditions:
468 + * The above copyright notice and this permission notice (including the next
469 + * paragraph) shall be included in all copies or substantial portions of the
472 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
473 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
474 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
475 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
476 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
477 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
478 + * DEALINGS IN THE SOFTWARE.
481 +#ifndef PIPEWIRE_EXT_SESSION_MANAGER_INTERFACES_H
482 +#define PIPEWIRE_EXT_SESSION_MANAGER_INTERFACES_H
484 +#include <spa/utils/defs.h>
485 +#include <spa/utils/hook.h>
487 +#include "introspect.h"
493 +#define PW_VERSION_SESSION_PROXY 0
494 +struct pw_session_proxy { struct spa_interface iface; };
495 +#define PW_VERSION_ENDPOINT_PROXY 0
496 +struct pw_endpoint_proxy { struct spa_interface iface; };
497 +#define PW_VERSION_ENDPOINT_STREAM_PROXY 0
498 +struct pw_endpoint_stream_proxy { struct spa_interface iface; };
499 +#define PW_VERSION_ENDPOINT_LINK_PROXY 0
500 +struct pw_endpoint_link_proxy { struct spa_interface iface; };
504 +#define PW_SESSION_PROXY_EVENT_INFO 0
505 +#define PW_SESSION_PROXY_EVENT_PARAM 1
506 +#define PW_SESSION_PROXY_EVENT_NUM 2
508 +struct pw_session_proxy_events {
509 +#define PW_VERSION_SESSION_PROXY_EVENTS 0
510 + uint32_t version; /**< version of this structure */
513 + * Notify session info
515 + * \param info info about the session
517 + void (*info) (void *object, const struct pw_session_info *info);
520 + * Notify a session param
522 + * Event emited as a result of the enum_params method.
524 + * \param seq the sequence number of the request
525 + * \param id the param id
526 + * \param index the param index
527 + * \param next the param index of the next param
528 + * \param param the parameter
530 + void (*param) (void *object, int seq,
531 + uint32_t id, uint32_t index, uint32_t next,
532 + const struct spa_pod *param);
535 +#define PW_SESSION_PROXY_METHOD_ADD_LISTENER 0
536 +#define PW_SESSION_PROXY_METHOD_SUBSCRIBE_PARAMS 1
537 +#define PW_SESSION_PROXY_METHOD_ENUM_PARAMS 2
538 +#define PW_SESSION_PROXY_METHOD_SET_PARAM 3
539 +#define PW_SESSION_PROXY_METHOD_CREATE_LINK 4
540 +#define PW_SESSION_PROXY_METHOD_NUM 5
542 +struct pw_session_proxy_methods {
543 +#define PW_VERSION_SESSION_PROXY_METHODS 0
544 + uint32_t version; /**< version of this structure */
546 + int (*add_listener) (void *object,
547 + struct spa_hook *listener,
548 + const struct pw_session_proxy_events *events,
552 + * Subscribe to parameter changes
554 + * Automatically emit param events for the given ids when
555 + * they are changed.
557 + * \param ids an array of param ids
558 + * \param n_ids the number of ids in \a ids
560 + int (*subscribe_params) (void *object, uint32_t *ids, uint32_t n_ids);
563 + * Enumerate session parameters
565 + * Start enumeration of session parameters. For each param, a
566 + * param event will be emited.
568 + * \param seq a sequence number returned in the reply
569 + * \param id the parameter id to enumerate
570 + * \param start the start index or 0 for the first param
571 + * \param num the maximum number of params to retrieve
572 + * \param filter a param filter or NULL
574 + int (*enum_params) (void *object, int seq,
575 + uint32_t id, uint32_t start, uint32_t num,
576 + const struct spa_pod *filter);
579 + * Set a parameter on the session
581 + * \param id the parameter id to set
582 + * \param flags extra parameter flags
583 + * \param param the parameter to set
585 + int (*set_param) (void *object, uint32_t id, uint32_t flags,
586 + const struct spa_pod *param);
588 + int (*create_link) (void *object, const struct spa_dict *props);
591 +#define pw_session_proxy_method(o,method,version,...) \
593 + int _res = -ENOTSUP; \
594 + struct pw_session_proxy *_p = o; \
595 + spa_interface_call_res(&_p->iface, \
596 + struct pw_session_proxy_methods, _res, \
597 + method, version, ##__VA_ARGS__); \
601 +#define pw_session_proxy_add_listener(c,...) pw_session_proxy_method(c,add_listener,0,__VA_ARGS__)
602 +#define pw_session_proxy_subscribe_params(c,...) pw_session_proxy_method(c,subscribe_params,0,__VA_ARGS__)
603 +#define pw_session_proxy_enum_params(c,...) pw_session_proxy_method(c,enum_params,0,__VA_ARGS__)
604 +#define pw_session_proxy_create_link(c,...) pw_session_proxy_method(c,create_link,0,__VA_ARGS__)
608 +#define PW_ENDPOINT_PROXY_EVENT_INFO 0
609 +#define PW_ENDPOINT_PROXY_EVENT_PARAM 1
610 +#define PW_ENDPOINT_PROXY_EVENT_NUM 2
612 +struct pw_endpoint_proxy_events {
613 +#define PW_VERSION_ENDPOINT_PROXY_EVENTS 0
614 + uint32_t version; /**< version of this structure */
617 + * Notify endpoint info
619 + * \param info info about the endpoint
621 + void (*info) (void *object, const struct pw_endpoint_info *info);
624 + * Notify a endpoint param
626 + * Event emited as a result of the enum_params method.
628 + * \param seq the sequence number of the request
629 + * \param id the param id
630 + * \param index the param index
631 + * \param next the param index of the next param
632 + * \param param the parameter
634 + void (*param) (void *object, int seq,
635 + uint32_t id, uint32_t index, uint32_t next,
636 + const struct spa_pod *param);
639 +#define PW_ENDPOINT_PROXY_METHOD_ADD_LISTENER 0
640 +#define PW_ENDPOINT_PROXY_METHOD_SUBSCRIBE_PARAMS 1
641 +#define PW_ENDPOINT_PROXY_METHOD_ENUM_PARAMS 2
642 +#define PW_ENDPOINT_PROXY_METHOD_SET_PARAM 3
643 +#define PW_ENDPOINT_PROXY_METHOD_NUM 4
645 +struct pw_endpoint_proxy_methods {
646 +#define PW_VERSION_ENDPOINT_PROXY_METHODS 0
647 + uint32_t version; /**< version of this structure */
649 + int (*add_listener) (void *object,
650 + struct spa_hook *listener,
651 + const struct pw_endpoint_proxy_events *events,
655 + * Subscribe to parameter changes
657 + * Automatically emit param events for the given ids when
658 + * they are changed.
660 + * \param ids an array of param ids
661 + * \param n_ids the number of ids in \a ids
663 + int (*subscribe_params) (void *object, uint32_t *ids, uint32_t n_ids);
666 + * Enumerate endpoint parameters
668 + * Start enumeration of endpoint parameters. For each param, a
669 + * param event will be emited.
671 + * \param seq a sequence number returned in the reply
672 + * \param id the parameter id to enumerate
673 + * \param start the start index or 0 for the first param
674 + * \param num the maximum number of params to retrieve
675 + * \param filter a param filter or NULL
677 + int (*enum_params) (void *object, int seq,
678 + uint32_t id, uint32_t start, uint32_t num,
679 + const struct spa_pod *filter);
682 + * Set a parameter on the endpoint
684 + * \param id the parameter id to set
685 + * \param flags extra parameter flags
686 + * \param param the parameter to set
688 + int (*set_param) (void *object, uint32_t id, uint32_t flags,
689 + const struct spa_pod *param);
692 +#define pw_endpoint_proxy_method(o,method,version,...) \
694 + int _res = -ENOTSUP; \
695 + struct pw_endpoint_proxy *_p = o; \
696 + spa_interface_call_res(&_p->iface, \
697 + struct pw_endpoint_proxy_methods, _res, \
698 + method, version, ##__VA_ARGS__); \
702 +#define pw_endpoint_proxy_add_listener(c,...) pw_endpoint_proxy_method(c,add_listener,0,__VA_ARGS__)
703 +#define pw_endpoint_proxy_subscribe_params(c,...) pw_endpoint_proxy_method(c,subscribe_params,0,__VA_ARGS__)
704 +#define pw_endpoint_proxy_enum_params(c,...) pw_endpoint_proxy_method(c,enum_params,0,__VA_ARGS__)
706 +/* Endpoint Stream */
708 +#define PW_ENDPOINT_STREAM_PROXY_EVENT_INFO 0
709 +#define PW_ENDPOINT_STREAM_PROXY_EVENT_PARAM 1
710 +#define PW_ENDPOINT_STREAM_PROXY_EVENT_NUM 2
712 +struct pw_endpoint_stream_proxy_events {
713 +#define PW_VERSION_ENDPOINT_STREAM_PROXY_EVENTS 0
714 + uint32_t version; /**< version of this structure */
717 + * Notify endpoint stream info
719 + * \param info info about the endpoint stream
721 + void (*info) (void *object, const struct pw_endpoint_stream_info *info);
724 + * Notify a endpoint stream param
726 + * Event emited as a result of the enum_params method.
728 + * \param seq the sequence number of the request
729 + * \param id the param id
730 + * \param index the param index
731 + * \param next the param index of the next param
732 + * \param param the parameter
734 + void (*param) (void *object, int seq,
735 + uint32_t id, uint32_t index, uint32_t next,
736 + const struct spa_pod *param);
739 +#define PW_ENDPOINT_STREAM_PROXY_METHOD_ADD_LISTENER 0
740 +#define PW_ENDPOINT_STREAM_PROXY_METHOD_SUBSCRIBE_PARAMS 1
741 +#define PW_ENDPOINT_STREAM_PROXY_METHOD_ENUM_PARAMS 2
742 +#define PW_ENDPOINT_STREAM_PROXY_METHOD_SET_PARAM 3
743 +#define PW_ENDPOINT_STREAM_PROXY_METHOD_NUM 4
745 +struct pw_endpoint_stream_proxy_methods {
746 +#define PW_VERSION_ENDPOINT_STREAM_PROXY_METHODS 0
747 + uint32_t version; /**< version of this structure */
749 + int (*add_listener) (void *object,
750 + struct spa_hook *listener,
751 + const struct pw_endpoint_stream_proxy_events *events,
755 + * Subscribe to parameter changes
757 + * Automatically emit param events for the given ids when
758 + * they are changed.
760 + * \param ids an array of param ids
761 + * \param n_ids the number of ids in \a ids
763 + int (*subscribe_params) (void *object, uint32_t *ids, uint32_t n_ids);
766 + * Enumerate stream parameters
768 + * Start enumeration of stream parameters. For each param, a
769 + * param event will be emited.
771 + * \param seq a sequence number returned in the reply
772 + * \param id the parameter id to enumerate
773 + * \param start the start index or 0 for the first param
774 + * \param num the maximum number of params to retrieve
775 + * \param filter a param filter or NULL
777 + int (*enum_params) (void *object, int seq,
778 + uint32_t id, uint32_t start, uint32_t num,
779 + const struct spa_pod *filter);
782 + * Set a parameter on the stream
784 + * \param id the parameter id to set
785 + * \param flags extra parameter flags
786 + * \param param the parameter to set
788 + int (*set_param) (void *object, uint32_t id, uint32_t flags,
789 + const struct spa_pod *param);
792 +#define pw_endpoint_stream_proxy_method(o,method,version,...) \
794 + int _res = -ENOTSUP; \
795 + struct pw_endpoint_stream_proxy *_p = o; \
796 + spa_interface_call_res(&_p->iface, \
797 + struct pw_endpoint_stream_proxy_methods, _res, \
798 + method, version, ##__VA_ARGS__); \
802 +#define pw_endpoint_stream_proxy_add_listener(c,...) pw_endpoint_stream_proxy_method(c,add_listener,0,__VA_ARGS__)
803 +#define pw_endpoint_stream_proxy_subscribe_params(c,...) pw_endpoint_stream_proxy_method(c,subscribe_params,0,__VA_ARGS__)
804 +#define pw_endpoint_stream_proxy_enum_params(c,...) pw_endpoint_stream_proxy_method(c,enum_params,0,__VA_ARGS__)
808 +#define PW_ENDPOINT_LINK_PROXY_EVENT_INFO 0
809 +#define PW_ENDPOINT_LINK_PROXY_EVENT_PARAM 1
810 +#define PW_ENDPOINT_LINK_PROXY_EVENT_NUM 2
812 +struct pw_endpoint_link_proxy_events {
813 +#define PW_VERSION_ENDPOINT_LINK_PROXY_EVENTS 0
814 + uint32_t version; /**< version of this structure */
817 + * Notify endpoint link info
819 + * \param info info about the endpoint link
821 + void (*info) (void *object, const struct pw_endpoint_link_info *info);
824 + * Notify a endpoint link param
826 + * Event emited as a result of the enum_params method.
828 + * \param seq the sequence number of the request
829 + * \param id the param id
830 + * \param index the param index
831 + * \param next the param index of the next param
832 + * \param param the parameter
834 + void (*param) (void *object, int seq,
835 + uint32_t id, uint32_t index, uint32_t next,
836 + const struct spa_pod *param);
839 +#define PW_ENDPOINT_LINK_PROXY_METHOD_ADD_LISTENER 0
840 +#define PW_ENDPOINT_LINK_PROXY_METHOD_SUBSCRIBE_PARAMS 1
841 +#define PW_ENDPOINT_LINK_PROXY_METHOD_ENUM_PARAMS 2
842 +#define PW_ENDPOINT_LINK_PROXY_METHOD_SET_PARAM 3
843 +#define PW_ENDPOINT_LINK_PROXY_METHOD_REQUEST_STATE 4
844 +#define PW_ENDPOINT_LINK_PROXY_METHOD_DESTROY 5
845 +#define PW_ENDPOINT_LINK_PROXY_METHOD_NUM 6
847 +struct pw_endpoint_link_proxy_methods {
848 +#define PW_VERSION_ENDPOINT_LINK_PROXY_METHODS 0
849 + uint32_t version; /**< version of this structure */
851 + int (*add_listener) (void *object,
852 + struct spa_hook *listener,
853 + const struct pw_endpoint_link_proxy_events *events,
857 + * Subscribe to parameter changes
859 + * Automatically emit param events for the given ids when
860 + * they are changed.
862 + * \param ids an array of param ids
863 + * \param n_ids the number of ids in \a ids
865 + int (*subscribe_params) (void *object, uint32_t *ids, uint32_t n_ids);
868 + * Enumerate link parameters
870 + * Start enumeration of link parameters. For each param, a
871 + * param event will be emited.
873 + * \param seq a sequence number returned in the reply
874 + * \param id the parameter id to enumerate
875 + * \param start the start index or 0 for the first param
876 + * \param num the maximum number of params to retrieve
877 + * \param filter a param filter or NULL
879 + int (*enum_params) (void *object, int seq,
880 + uint32_t id, uint32_t start, uint32_t num,
881 + const struct spa_pod *filter);
884 + * Set a parameter on the link
886 + * \param id the parameter id to set
887 + * \param flags extra parameter flags
888 + * \param param the parameter to set
890 + int (*set_param) (void *object, uint32_t id, uint32_t flags,
891 + const struct spa_pod *param);
893 + int (*request_state) (void *object, enum pw_endpoint_link_state state);
895 + int (*destroy) (void *object);
899 +#define pw_endpoint_link_proxy_method(o,method,version,...) \
901 + int _res = -ENOTSUP; \
902 + struct pw_endpoint_link_proxy *_p = o; \
903 + spa_interface_call_res(&_p->iface, \
904 + struct pw_endpoint_link_proxy_methods, _res, \
905 + method, version, ##__VA_ARGS__); \
909 +#define pw_endpoint_link_proxy_add_listener(c,...) pw_endpoint_link_proxy_method(c,add_listener,0,__VA_ARGS__)
910 +#define pw_endpoint_link_proxy_subscribe_params(c,...) pw_endpoint_link_proxy_method(c,subscribe_params,0,__VA_ARGS__)
911 +#define pw_endpoint_link_proxy_enum_params(c,...) pw_endpoint_link_proxy_method(c,enum_params,0,__VA_ARGS__)
912 +#define pw_endpoint_link_proxy_request_state(c,...) pw_endpoint_link_proxy_method(c,request_state,0,__VA_ARGS__)
913 +#define pw_endpoint_link_proxy_destroy(c,...) pw_endpoint_link_proxy_method(c,destroy,0,__VA_ARGS__)
920 +#endif /* PIPEWIRE_EXT_SESSION_MANAGER_INTERFACES_H */
921 diff --git a/src/extensions/session-manager/introspect.h b/src/extensions/session-manager/introspect.h
923 index 00000000..3b0e4113
925 +++ b/src/extensions/session-manager/introspect.h
929 + * Copyright © 2019 Collabora Ltd.
930 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
932 + * Permission is hereby granted, free of charge, to any person obtaining a
933 + * copy of this software and associated documentation files (the "Software"),
934 + * to deal in the Software without restriction, including without limitation
935 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
936 + * and/or sell copies of the Software, and to permit persons to whom the
937 + * Software is furnished to do so, subject to the following conditions:
939 + * The above copyright notice and this permission notice (including the next
940 + * paragraph) shall be included in all copies or substantial portions of the
943 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
944 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
945 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
946 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
947 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
948 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
949 + * DEALINGS IN THE SOFTWARE.
952 +#ifndef PIPEWIRE_EXT_SESSION_MANAGER_INTROSPECT_H
953 +#define PIPEWIRE_EXT_SESSION_MANAGER_INTROSPECT_H
955 +#include <spa/utils/defs.h>
956 +#include <spa/utils/dict.h>
957 +#include <spa/param/param.h>
963 +#define PW_KEY_ENDPOINT_ID "endpoint.id"
964 +#define PW_KEY_SESSION_ID "session.id"
966 +enum pw_endpoint_direction {
967 + PW_ENDPOINT_DIRECTION_SINK_INPUT = SPA_DIRECTION_INPUT,
968 + PW_ENDPOINT_DIRECTION_SOURCE_OUTPUT = SPA_DIRECTION_OUTPUT,
969 + PW_ENDPOINT_DIRECTION_SOURCE,
970 + PW_ENDPOINT_DIRECTION_SINK,
973 +enum pw_endpoint_link_state {
974 + PW_ENDPOINT_LINK_STATE_ERROR = -1,
975 + PW_ENDPOINT_LINK_STATE_PREPARING,
976 + PW_ENDPOINT_LINK_STATE_INACTIVE,
977 + PW_ENDPOINT_LINK_STATE_ACTIVE,
980 +struct pw_session_info {
981 +#define PW_VERSION_SESSION_INFO 0
982 + uint32_t version; /**< version of this structure */
983 + uint32_t id; /**< the session id (global) */
984 +#define PW_SESSION_CHANGE_MASK_PROPS (1 << 0)
985 +#define PW_SESSION_CHANGE_MASK_PARAMS (1 << 1)
986 +#define PW_SESSION_CHANGE_MASK_ALL ((1 << 2)-1)
987 + uint32_t change_mask; /**< bitfield of changed fields since last call */
988 + struct spa_dict *props; /**< extra properties */
989 + struct spa_param_info *params; /**< parameters */
990 + uint32_t n_params; /**< number of items in \a params */
993 +struct pw_endpoint_info {
994 +#define PW_VERSION_ENDPOINT_INFO 0
995 + uint32_t version; /**< version of this structure */
996 + uint32_t id; /**< the endpoint id (global) */
997 + char *name; /**< name of the endpoint */
998 + char *media_class; /**< media class of the endpoint */
999 + enum pw_endpoint_direction direction; /**< direction of the endpoint */
1000 +#define PW_ENDPOINT_FLAG_PROVIDES_SESSION (1 << 0)
1001 + uint32_t flags; /**< additional flags */
1002 +#define PW_ENDPOINT_CHANGE_MASK_STREAMS (1 << 0)
1003 +#define PW_ENDPOINT_CHANGE_MASK_SESSION (1 << 1)
1004 +#define PW_ENDPOINT_CHANGE_MASK_PROPS (1 << 2)
1005 +#define PW_ENDPOINT_CHANGE_MASK_PARAMS (1 << 3)
1006 +#define PW_ENDPOINT_CHANGE_MASK_ALL ((1 << 4)-1)
1007 + uint32_t change_mask; /**< bitfield of changed fields since last call */
1008 + uint32_t n_streams; /**< number of streams available */
1009 + uint32_t session_id; /**< the id of the controlling session */
1010 + struct spa_dict *props; /**< extra properties */
1011 + struct spa_param_info *params; /**< parameters */
1012 + uint32_t n_params; /**< number of items in \a params */
1015 +struct pw_endpoint_stream_info {
1016 +#define PW_VERSION_ENDPOINT_STREAM_INFO 0
1017 + uint32_t version; /**< version of this structure */
1018 + uint32_t id; /**< the stream id (local or global) */
1019 + uint32_t endpoint_id; /**< the endpoint id (global) */
1020 + char *name; /**< name of the stream */
1021 +#define PW_ENDPOINT_STREAM_CHANGE_MASK_LINK_PARAMS (1 << 0)
1022 +#define PW_ENDPOINT_STREAM_CHANGE_MASK_PROPS (1 << 1)
1023 +#define PW_ENDPOINT_STREAM_CHANGE_MASK_PARAMS (1 << 2)
1024 +#define PW_ENDPOINT_STREAM_CHANGE_MASK_ALL ((1 << 3)-1)
1025 + uint32_t change_mask; /**< bitfield of changed fields since last call */
1026 + struct spa_pod *link_params; /**< information for linking this stream */
1027 + struct spa_dict *props; /**< extra properties */
1028 + struct spa_param_info *params; /**< parameters */
1029 + uint32_t n_params; /**< number of items in \a params */
1032 +struct pw_endpoint_link_info {
1033 +#define PW_VERSION_ENDPOINT_LINK_INFO 0
1034 + uint32_t version; /**< version of this structure */
1035 + uint32_t id; /**< the link id (global) */
1036 + uint32_t session_id; /**< the session id (global) */
1037 + uint32_t output_endpoint_id; /**< the output endpoint id (global) */
1038 + uint32_t output_stream_id; /**< the output stream id (local or global) */
1039 + uint32_t input_endpoint_id; /**< the input endpoint id (global) */
1040 + uint32_t input_stream_id; /**< the input stream id (local or global) */
1041 +#define PW_ENDPOINT_LINK_CHANGE_MASK_STATE (1 << 0)
1042 +#define PW_ENDPOINT_LINK_CHANGE_MASK_PROPS (1 << 1)
1043 +#define PW_ENDPOINT_LINK_CHANGE_MASK_PARAMS (1 << 2)
1044 +#define PW_ENDPOINT_LINK_CHANGE_MASK_ALL ((1 << 3)-1)
1045 + uint32_t change_mask; /**< bitfield of changed fields since last call */
1046 + enum pw_endpoint_link_state state; /**< the state of the link */
1047 + char *error; /**< error string if state == ERROR */
1048 + struct spa_dict *props; /**< extra properties */
1049 + struct spa_param_info *params; /**< parameters */
1050 + uint32_t n_params; /**< number of items in \a params */
1057 +#endif /* PIPEWIRE_EXT_SESSION_MANAGER_INTROSPECT_H */
1058 diff --git a/src/extensions/session-manager/keys.h b/src/extensions/session-manager/keys.h
1059 new file mode 100644
1060 index 00000000..a7167510
1062 +++ b/src/extensions/session-manager/keys.h
1066 + * Copyright © 2019 Collabora Ltd.
1067 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1069 + * Permission is hereby granted, free of charge, to any person obtaining a
1070 + * copy of this software and associated documentation files (the "Software"),
1071 + * to deal in the Software without restriction, including without limitation
1072 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1073 + * and/or sell copies of the Software, and to permit persons to whom the
1074 + * Software is furnished to do so, subject to the following conditions:
1076 + * The above copyright notice and this permission notice (including the next
1077 + * paragraph) shall be included in all copies or substantial portions of the
1080 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1081 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1082 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1083 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1084 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1085 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1086 + * DEALINGS IN THE SOFTWARE.
1089 +#ifndef PIPEWIRE_EXT_SESSION_MANAGER_KEYS_H
1090 +#define PIPEWIRE_EXT_SESSION_MANAGER_KEYS_H
1096 +#define PW_KEY_ENDPOINT_ID "endpoint.id"
1097 +#define PW_KEY_SESSION_ID "session.id"
1103 +#endif /* PIPEWIRE_EXT_SESSION_MANAGER_KEYS_H */
1104 diff --git a/src/modules/meson.build b/src/modules/meson.build
1105 index 9e1e94bd..da2684b6 100644
1106 --- a/src/modules/meson.build
1107 +++ b/src/modules/meson.build
1108 @@ -100,3 +100,20 @@ pipewire_module_adapter = shared_library('pipewire-module-adapter',
1109 install_dir : modules_install_dir,
1110 dependencies : [mathlib, dl_lib, rt_lib, pipewire_dep],
1113 +pipewire_module_session_manager = shared_library('pipewire-module-session-manager',
1114 + [ 'module-session-manager.c',
1115 + 'module-session-manager/client-endpoint.c',
1116 + 'module-session-manager/client-session.c',
1117 + 'module-session-manager/endpoint-link.c',
1118 + 'module-session-manager/endpoint-stream.c',
1119 + 'module-session-manager/endpoint.c',
1120 + 'module-session-manager/session.c',
1121 + 'module-session-manager/protocol-native.c',
1123 + c_args : pipewire_module_c_args,
1124 + include_directories : [configinc, spa_inc],
1126 + install_dir : modules_install_dir,
1127 + dependencies : [mathlib, dl_lib, pipewire_dep],
1129 diff --git a/src/modules/module-session-manager.c b/src/modules/module-session-manager.c
1130 new file mode 100644
1131 index 00000000..dbea3357
1133 +++ b/src/modules/module-session-manager.c
1137 + * Copyright © 2019 Collabora Ltd.
1138 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1140 + * Permission is hereby granted, free of charge, to any person obtaining a
1141 + * copy of this software and associated documentation files (the "Software"),
1142 + * to deal in the Software without restriction, including without limitation
1143 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1144 + * and/or sell copies of the Software, and to permit persons to whom the
1145 + * Software is furnished to do so, subject to the following conditions:
1147 + * The above copyright notice and this permission notice (including the next
1148 + * paragraph) shall be included in all copies or substantial portions of the
1151 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1152 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1153 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1154 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1155 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1156 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1157 + * DEALINGS IN THE SOFTWARE.
1160 +#include "config.h"
1162 +#include <pipewire/pipewire.h>
1164 +/* client-endpoint.c */
1165 +int client_endpoint_factory_init(struct pw_module *module);
1166 +/* client-session.c */
1167 +int client_session_factory_init(struct pw_module *module);
1168 +/* protocol-native.c */
1169 +struct pw_protocol *pw_protocol_native_ext_session_manager_init(struct pw_core *core);
1171 +static const struct spa_dict_item module_props[] = {
1172 + { PW_KEY_MODULE_AUTHOR, "George Kiagiadakis <george.kiagiadakis@collabora.com>" },
1173 + { PW_KEY_MODULE_DESCRIPTION, "Implements objects for session management" },
1174 + { PW_KEY_MODULE_VERSION, PACKAGE_VERSION },
1178 +int pipewire__module_init(struct pw_module *module, const char *args)
1180 + struct pw_core *core = pw_module_get_core(module);
1182 + client_endpoint_factory_init(module);
1183 + client_session_factory_init(module);
1185 + pw_protocol_native_ext_session_manager_init(core);
1187 + pw_module_update_properties(module, &SPA_DICT_INIT_ARRAY(module_props));
1191 diff --git a/src/modules/module-session-manager/client-endpoint.c b/src/modules/module-session-manager/client-endpoint.c
1192 new file mode 100644
1193 index 00000000..0e501c9f
1195 +++ b/src/modules/module-session-manager/client-endpoint.c
1199 + * Copyright © 2019 Collabora Ltd.
1200 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1202 + * Permission is hereby granted, free of charge, to any person obtaining a
1203 + * copy of this software and associated documentation files (the "Software"),
1204 + * to deal in the Software without restriction, including without limitation
1205 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1206 + * and/or sell copies of the Software, and to permit persons to whom the
1207 + * Software is furnished to do so, subject to the following conditions:
1209 + * The above copyright notice and this permission notice (including the next
1210 + * paragraph) shall be included in all copies or substantial portions of the
1213 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1214 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1215 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1216 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1217 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1218 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1219 + * DEALINGS IN THE SOFTWARE.
1222 +#include <stdbool.h>
1223 +#include <string.h>
1225 +#include <pipewire/pipewire.h>
1226 +#include <extensions/session-manager.h>
1228 +#include "client-endpoint.h"
1229 +#include "endpoint.h"
1230 +#include "endpoint-stream.h"
1232 +#include <pipewire/private.h>
1234 +#define NAME "client-endpoint"
1236 +struct factory_data {
1237 + struct pw_factory *factory;
1238 + struct pw_module *module;
1239 + struct spa_hook module_listener;
1242 +static struct endpoint_stream *find_stream(struct client_endpoint *this, uint32_t id)
1244 + struct endpoint_stream *s;
1245 + spa_list_for_each(s, &this->streams, link) {
1252 +static int client_endpoint_update(void *object,
1253 + uint32_t change_mask,
1254 + uint32_t n_params,
1255 + const struct spa_pod **params,
1256 + const struct pw_endpoint_info *info)
1258 + struct client_endpoint *this = object;
1259 + struct endpoint *endpoint = &this->endpoint;
1261 + return endpoint_update(endpoint, change_mask, n_params, params, info);
1264 +static int client_endpoint_stream_update(void *object,
1265 + uint32_t stream_id,
1266 + uint32_t change_mask,
1267 + uint32_t n_params,
1268 + const struct spa_pod **params,
1269 + const struct pw_endpoint_stream_info *info)
1271 + struct client_endpoint *this = object;
1272 + struct endpoint *endpoint = &this->endpoint;
1273 + struct endpoint_stream *stream = find_stream(this, stream_id);
1274 + struct pw_properties *props = NULL;
1277 + struct pw_core *core = pw_global_get_core(endpoint->global);
1278 + const char *keys[] = {
1279 + PW_KEY_FACTORY_ID,
1281 + PW_KEY_ENDPOINT_ID,
1285 + stream = calloc(1, sizeof(struct endpoint_stream));
1289 + props = pw_properties_new(NULL, NULL);
1292 + pw_properties_copy_keys (endpoint->props, props, keys);
1294 + if (endpoint_stream_init(stream, stream_id, endpoint->info.id,
1295 + this, core, props) < 0)
1298 + spa_list_append(&this->streams, &stream->link);
1300 + else if (change_mask & PW_CLIENT_ENDPOINT_STREAM_UPDATE_DESTROYED) {
1301 + endpoint_stream_clear(stream);
1302 + spa_list_remove(&stream->link);
1308 + endpoint_stream_update(stream, change_mask, n_params, params, info)
1313 + pw_properties_free(props);
1315 + pw_log_error(NAME" %p: cannot update stream: no memory", this);
1316 + pw_resource_error(this->resource, -ENOMEM,
1317 + NAME" %p: cannot update stream: no memory", this);
1321 +static struct pw_client_endpoint_proxy_methods methods = {
1322 + PW_VERSION_CLIENT_ENDPOINT_PROXY_METHODS,
1323 + .update = client_endpoint_update,
1324 + .stream_update = client_endpoint_stream_update,
1327 +static void client_endpoint_destroy(void *data)
1329 + struct client_endpoint *this = data;
1330 + struct endpoint_stream *s;
1332 + pw_log_debug(NAME" %p: destroy", this);
1334 + spa_list_consume(s, &this->streams, link) {
1335 + endpoint_stream_clear(s);
1336 + spa_list_remove(&s->link);
1339 + endpoint_clear(&this->endpoint);
1340 + spa_hook_remove(&this->resource_listener);
1345 +static const struct pw_resource_events resource_events = {
1346 + PW_VERSION_RESOURCE_EVENTS,
1347 + .destroy = client_endpoint_destroy,
1350 +static void *create_object(void *data,
1351 + struct pw_resource *owner_resource,
1354 + struct pw_properties *properties,
1357 + struct factory_data *d = data;
1358 + struct pw_factory *factory = d->factory;
1359 + struct client_endpoint *this;
1360 + struct pw_client *owner = pw_resource_get_client(owner_resource);
1361 + struct pw_core *core = pw_client_get_core(owner);
1363 + this = calloc(1, sizeof(struct client_endpoint));
1367 + spa_list_init(&this->streams);
1369 + pw_log_debug(NAME" %p: new", this);
1372 + properties = pw_properties_new(NULL, NULL);
1376 + pw_properties_setf(properties, PW_KEY_CLIENT_ID, "%d", owner->global->id);
1377 + pw_properties_setf(properties, PW_KEY_FACTORY_ID, "%d", factory->global->id);
1379 + this->resource = pw_resource_new(owner, new_id, PW_PERM_RWX, type, version, 0);
1380 + if (this->resource == NULL)
1383 + if (endpoint_init(&this->endpoint, this, core, properties) < 0)
1386 + pw_resource_add_listener(this->resource, &this->resource_listener,
1387 + &resource_events, this);
1388 + pw_resource_add_object_listener(this->resource, &this->object_listener,
1395 + pw_properties_free(properties);
1396 + if (this && this->resource)
1397 + pw_resource_destroy(this->resource);
1399 + pw_log_error("can't create client endpoint: no memory");
1400 + pw_resource_error(owner_resource, -ENOMEM,
1401 + "can't create client endpoint: no memory");
1405 +static const struct pw_factory_implementation impl_factory = {
1406 + PW_VERSION_FACTORY_IMPLEMENTATION,
1407 + .create_object = create_object,
1410 +static void module_destroy(void *data)
1412 + struct factory_data *d = data;
1414 + spa_hook_remove(&d->module_listener);
1415 + pw_factory_destroy(d->factory);
1418 +static void module_registered(void *data)
1420 + struct factory_data *d = data;
1421 + struct pw_module *module = d->module;
1422 + struct pw_factory *factory = d->factory;
1423 + struct spa_dict_item items[1];
1427 + snprintf(id, sizeof(id), "%d", module->global->id);
1428 + items[0] = SPA_DICT_ITEM_INIT(PW_KEY_MODULE_ID, id);
1429 + pw_factory_update_properties(factory, &SPA_DICT_INIT(items, 1));
1431 + if ((res = pw_factory_register(factory, NULL)) < 0) {
1432 + pw_log_error(NAME" %p: can't register factory: %s", factory, spa_strerror(res));
1436 +static const struct pw_module_events module_events = {
1437 + PW_VERSION_MODULE_EVENTS,
1438 + .destroy = module_destroy,
1439 + .registered = module_registered,
1442 +int client_endpoint_factory_init(struct pw_module *module)
1444 + struct pw_core *core = pw_module_get_core(module);
1445 + struct pw_factory *factory;
1446 + struct factory_data *data;
1448 + factory = pw_factory_new(core,
1449 + "client-endpoint",
1450 + PW_TYPE_INTERFACE_ClientEndpoint,
1451 + PW_VERSION_CLIENT_ENDPOINT_PROXY,
1454 + if (factory == NULL)
1457 + data = pw_factory_get_user_data(factory);
1458 + data->factory = factory;
1459 + data->module = module;
1461 + pw_factory_set_implementation(factory, &impl_factory, data);
1463 + pw_module_add_listener(module, &data->module_listener, &module_events, data);
1467 diff --git a/src/modules/module-session-manager/client-endpoint.h b/src/modules/module-session-manager/client-endpoint.h
1468 new file mode 100644
1469 index 00000000..394e9fa8
1471 +++ b/src/modules/module-session-manager/client-endpoint.h
1475 + * Copyright © 2019 Collabora Ltd.
1476 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1478 + * Permission is hereby granted, free of charge, to any person obtaining a
1479 + * copy of this software and associated documentation files (the "Software"),
1480 + * to deal in the Software without restriction, including without limitation
1481 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1482 + * and/or sell copies of the Software, and to permit persons to whom the
1483 + * Software is furnished to do so, subject to the following conditions:
1485 + * The above copyright notice and this permission notice (including the next
1486 + * paragraph) shall be included in all copies or substantial portions of the
1489 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1490 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1491 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1492 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1493 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1494 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1495 + * DEALINGS IN THE SOFTWARE.
1498 +#ifndef MODULE_SESSION_MANAGER_CLIENT_ENDPOINT_H
1499 +#define MODULE_SESSION_MANAGER_CLIENT_ENDPOINT_H
1501 +#include "endpoint.h"
1507 +struct client_endpoint {
1508 + struct pw_resource *resource;
1509 + struct spa_hook resource_listener;
1510 + struct spa_hook object_listener;
1511 + struct endpoint endpoint;
1512 + struct spa_list streams;
1515 +#define pw_client_endpoint_resource(r,m,v,...) \
1516 + pw_resource_call_res(r,struct pw_client_endpoint_proxy_events,m,v,__VA_ARGS__)
1517 +#define pw_client_endpoint_resource_set_id(r,...) \
1518 + pw_client_endpoint_resource(r,set_id,0,__VA_ARGS__)
1519 +#define pw_client_endpoint_resource_set_session_id(r,...) \
1520 + pw_client_endpoint_resource(r,set_session_id,0,__VA_ARGS__)
1521 +#define pw_client_endpoint_resource_set_param(r,...) \
1522 + pw_client_endpoint_resource(r,set_param,0,__VA_ARGS__)
1523 +#define pw_client_endpoint_resource_stream_set_param(r,...) \
1524 + pw_client_endpoint_resource(r,stream_set_param,0,__VA_ARGS__)
1526 +int client_endpoint_factory_init(struct pw_module *module);
1532 +#endif /* MODULE_SESSION_MANAGER_CLIENT_ENDPOINT_H */
1533 diff --git a/src/modules/module-session-manager/client-session.c b/src/modules/module-session-manager/client-session.c
1534 new file mode 100644
1535 index 00000000..9b20d833
1537 +++ b/src/modules/module-session-manager/client-session.c
1541 + * Copyright © 2019 Collabora Ltd.
1542 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1544 + * Permission is hereby granted, free of charge, to any person obtaining a
1545 + * copy of this software and associated documentation files (the "Software"),
1546 + * to deal in the Software without restriction, including without limitation
1547 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1548 + * and/or sell copies of the Software, and to permit persons to whom the
1549 + * Software is furnished to do so, subject to the following conditions:
1551 + * The above copyright notice and this permission notice (including the next
1552 + * paragraph) shall be included in all copies or substantial portions of the
1555 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1556 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1557 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1558 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1559 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1560 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1561 + * DEALINGS IN THE SOFTWARE.
1564 +#include <stdbool.h>
1565 +#include <string.h>
1567 +#include <pipewire/pipewire.h>
1568 +#include <extensions/session-manager.h>
1570 +#include "client-session.h"
1571 +#include "session.h"
1572 +#include "endpoint-link.h"
1574 +#include <pipewire/private.h>
1576 +#define NAME "client-session"
1578 +struct factory_data {
1579 + struct pw_factory *factory;
1580 + struct pw_module *module;
1581 + struct spa_hook module_listener;
1584 +static struct endpoint_link *find_link(struct client_session *this, uint32_t id)
1586 + struct endpoint_link *l;
1587 + spa_list_for_each(l, &this->links, link) {
1594 +static int client_session_update(void *object,
1595 + uint32_t change_mask,
1596 + uint32_t n_params,
1597 + const struct spa_pod **params,
1598 + const struct pw_session_info *info)
1600 + struct client_session *this = object;
1601 + struct session *session = &this->session;
1603 + return session_update(session, change_mask, n_params, params, info);
1606 +static int client_session_link_update(void *object,
1608 + uint32_t change_mask,
1609 + uint32_t n_params,
1610 + const struct spa_pod **params,
1611 + const struct pw_endpoint_link_info *info)
1613 + struct client_session *this = object;
1614 + struct session *session = &this->session;
1615 + struct endpoint_link *link = find_link(this, link_id);
1616 + struct pw_properties *props = NULL;
1619 + struct pw_core *core = pw_global_get_core(session->global);
1620 + const char *keys[] = {
1621 + PW_KEY_FACTORY_ID,
1623 + PW_KEY_SESSION_ID,
1627 + link = calloc(1, sizeof(struct endpoint_link));
1631 + props = pw_properties_new(NULL, NULL);
1634 + pw_properties_copy_keys (session->props, props, keys);
1636 + if (endpoint_link_init(link, link_id, session->info.id,
1637 + this, core, props) < 0)
1640 + spa_list_append(&this->links, &link->link);
1642 + else if (change_mask & PW_CLIENT_SESSION_LINK_UPDATE_DESTROYED) {
1643 + endpoint_link_clear(link);
1644 + spa_list_remove(&link->link);
1650 + endpoint_link_update(link, change_mask, n_params, params, info)
1655 + pw_properties_free(props);
1657 + pw_log_error(NAME" %p: cannot update link: no memory", this);
1658 + pw_resource_error(this->resource, -ENOMEM,
1659 + NAME" %p: cannot update link: no memory", this);
1663 +static struct pw_client_session_proxy_methods methods = {
1664 + PW_VERSION_CLIENT_SESSION_PROXY_METHODS,
1665 + .update = client_session_update,
1666 + .link_update = client_session_link_update,
1669 +static void client_session_destroy(void *data)
1671 + struct client_session *this = data;
1672 + struct endpoint_link *l;
1674 + pw_log_debug(NAME" %p: destroy", this);
1676 + spa_list_consume(l, &this->links, link) {
1677 + endpoint_link_clear(l);
1678 + spa_list_remove(&l->link);
1681 + session_clear(&this->session);
1682 + spa_hook_remove(&this->resource_listener);
1687 +static const struct pw_resource_events resource_events = {
1688 + PW_VERSION_RESOURCE_EVENTS,
1689 + .destroy = client_session_destroy,
1692 +static void *create_object(void *data,
1693 + struct pw_resource *owner_resource,
1696 + struct pw_properties *properties,
1699 + struct factory_data *d = data;
1700 + struct pw_factory *factory = d->factory;
1701 + struct client_session *this;
1702 + struct pw_client *owner = pw_resource_get_client(owner_resource);
1703 + struct pw_core *core = pw_client_get_core(owner);
1705 + this = calloc(1, sizeof(struct client_session));
1709 + spa_list_init(&this->links);
1711 + pw_log_debug(NAME" %p: new", this);
1714 + properties = pw_properties_new(NULL, NULL);
1718 + pw_properties_setf(properties, PW_KEY_CLIENT_ID, "%d", owner->global->id);
1719 + pw_properties_setf(properties, PW_KEY_FACTORY_ID, "%d", factory->global->id);
1721 + this->resource = pw_resource_new(owner, new_id, PW_PERM_RWX, type, version, 0);
1722 + if (this->resource == NULL)
1725 + if (session_init(&this->session, this, core, properties) < 0)
1728 + pw_resource_add_listener(this->resource, &this->resource_listener,
1729 + &resource_events, this);
1730 + pw_resource_add_object_listener(this->resource, &this->object_listener,
1737 + pw_properties_free(properties);
1738 + if (this && this->resource)
1739 + pw_resource_destroy(this->resource);
1741 + pw_log_error("can't create client session: no memory");
1742 + pw_resource_error(owner_resource, -ENOMEM,
1743 + "can't create client session: no memory");
1747 +static const struct pw_factory_implementation impl_factory = {
1748 + PW_VERSION_FACTORY_IMPLEMENTATION,
1749 + .create_object = create_object,
1752 +static void module_destroy(void *data)
1754 + struct factory_data *d = data;
1756 + spa_hook_remove(&d->module_listener);
1757 + pw_factory_destroy(d->factory);
1760 +static void module_registered(void *data)
1762 + struct factory_data *d = data;
1763 + struct pw_module *module = d->module;
1764 + struct pw_factory *factory = d->factory;
1765 + struct spa_dict_item items[1];
1769 + snprintf(id, sizeof(id), "%d", module->global->id);
1770 + items[0] = SPA_DICT_ITEM_INIT(PW_KEY_MODULE_ID, id);
1771 + pw_factory_update_properties(factory, &SPA_DICT_INIT(items, 1));
1773 + if ((res = pw_factory_register(factory, NULL)) < 0) {
1774 + pw_log_error(NAME" %p: can't register factory: %s", factory, spa_strerror(res));
1778 +static const struct pw_module_events module_events = {
1779 + PW_VERSION_MODULE_EVENTS,
1780 + .destroy = module_destroy,
1781 + .registered = module_registered,
1784 +int client_session_factory_init(struct pw_module *module)
1786 + struct pw_core *core = pw_module_get_core(module);
1787 + struct pw_factory *factory;
1788 + struct factory_data *data;
1790 + factory = pw_factory_new(core,
1792 + PW_TYPE_INTERFACE_ClientSession,
1793 + PW_VERSION_CLIENT_SESSION_PROXY,
1796 + if (factory == NULL)
1799 + data = pw_factory_get_user_data(factory);
1800 + data->factory = factory;
1801 + data->module = module;
1803 + pw_factory_set_implementation(factory, &impl_factory, data);
1805 + pw_module_add_listener(module, &data->module_listener, &module_events, data);
1809 diff --git a/src/modules/module-session-manager/client-session.h b/src/modules/module-session-manager/client-session.h
1810 new file mode 100644
1811 index 00000000..c764564d
1813 +++ b/src/modules/module-session-manager/client-session.h
1817 + * Copyright © 2019 Collabora Ltd.
1818 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1820 + * Permission is hereby granted, free of charge, to any person obtaining a
1821 + * copy of this software and associated documentation files (the "Software"),
1822 + * to deal in the Software without restriction, including without limitation
1823 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1824 + * and/or sell copies of the Software, and to permit persons to whom the
1825 + * Software is furnished to do so, subject to the following conditions:
1827 + * The above copyright notice and this permission notice (including the next
1828 + * paragraph) shall be included in all copies or substantial portions of the
1831 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1832 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1833 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1834 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1835 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1836 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1837 + * DEALINGS IN THE SOFTWARE.
1840 +#ifndef MODULE_SESSION_MANAGER_CLIENT_SESSION_H
1841 +#define MODULE_SESSION_MANAGER_CLIENT_SESSION_H
1843 +#include "session.h"
1849 +struct client_session {
1850 + struct pw_resource *resource;
1851 + struct spa_hook resource_listener;
1852 + struct spa_hook object_listener;
1853 + struct session session;
1854 + struct spa_list links;
1857 +#define pw_client_session_resource(r,m,v,...) \
1858 + pw_resource_call_res(r,struct pw_client_session_proxy_events,m,v,__VA_ARGS__)
1859 +#define pw_client_session_resource_set_id(r,...) \
1860 + pw_client_session_resource(r,set_id,0,__VA_ARGS__)
1861 +#define pw_client_session_resource_set_param(r,...) \
1862 + pw_client_session_resource(r,set_param,0,__VA_ARGS__)
1863 +#define pw_client_session_resource_link_set_param(r,...) \
1864 + pw_client_session_resource(r,link_set_param,0,__VA_ARGS__)
1865 +#define pw_client_session_resource_create_link(r,...) \
1866 + pw_client_session_resource(r,create_link,0,__VA_ARGS__)
1867 +#define pw_client_session_resource_destroy_link(r,...) \
1868 + pw_client_session_resource(r,destroy_link,0,__VA_ARGS__)
1869 +#define pw_client_session_resource_link_request_state(r,...) \
1870 + pw_client_session_resource(r,link_request_state,0,__VA_ARGS__)
1876 +#endif /* MODULE_SESSION_MANAGER_CLIENT_SESSION_H */
1877 diff --git a/src/modules/module-session-manager/endpoint-link.c b/src/modules/module-session-manager/endpoint-link.c
1878 new file mode 100644
1879 index 00000000..bce06598
1881 +++ b/src/modules/module-session-manager/endpoint-link.c
1885 + * Copyright © 2019 Collabora Ltd.
1886 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
1888 + * Permission is hereby granted, free of charge, to any person obtaining a
1889 + * copy of this software and associated documentation files (the "Software"),
1890 + * to deal in the Software without restriction, including without limitation
1891 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
1892 + * and/or sell copies of the Software, and to permit persons to whom the
1893 + * Software is furnished to do so, subject to the following conditions:
1895 + * The above copyright notice and this permission notice (including the next
1896 + * paragraph) shall be included in all copies or substantial portions of the
1899 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1900 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1901 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1902 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1903 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1904 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
1905 + * DEALINGS IN THE SOFTWARE.
1908 +#include <stdbool.h>
1909 +#include <string.h>
1911 +#include <pipewire/pipewire.h>
1912 +#include <extensions/session-manager.h>
1914 +#include <spa/pod/filter.h>
1916 +#include "endpoint-link.h"
1917 +#include "client-session.h"
1919 +#include <pipewire/private.h>
1921 +#define NAME "endpoint-link"
1923 +struct resource_data {
1924 + struct endpoint_link *link;
1925 + struct spa_hook resource_listener;
1926 + struct spa_hook object_listener;
1927 + uint32_t n_subscribe_ids;
1928 + uint32_t subscribe_ids[32];
1931 +#define pw_endpoint_link_resource(r,m,v,...) \
1932 + pw_resource_call(r,struct pw_endpoint_link_proxy_events,m,v,__VA_ARGS__)
1933 +#define pw_endpoint_link_resource_info(r,...) \
1934 + pw_endpoint_link_resource(r,info,0,__VA_ARGS__)
1935 +#define pw_endpoint_link_resource_param(r,...) \
1936 + pw_endpoint_link_resource(r,param,0,__VA_ARGS__)
1938 +static int endpoint_link_enum_params (void *object, int seq,
1939 + uint32_t id, uint32_t start, uint32_t num,
1940 + const struct spa_pod *filter)
1942 + struct pw_resource *resource = object;
1943 + struct resource_data *data = pw_resource_get_user_data(resource);
1944 + struct endpoint_link *this = data->link;
1945 + struct spa_pod *result;
1946 + struct spa_pod *param;
1947 + uint8_t buffer[1024];
1948 + struct spa_pod_builder b = { 0 };
1950 + uint32_t next = start;
1951 + uint32_t count = 0;
1955 + if (index >= this->n_params)
1958 + param = this->params[index];
1960 + if (param == NULL || !spa_pod_is_object_id(param, id))
1963 + spa_pod_builder_init(&b, buffer, sizeof(buffer));
1964 + if (spa_pod_filter(&b, &result, param, filter) != 0)
1967 + pw_log_debug(NAME" %p: %d param %u", this, seq, index);
1969 + pw_endpoint_link_resource_param(resource, seq, id, index, next, result);
1971 + if (++count == num)
1977 +static int endpoint_link_subscribe_params (void *object, uint32_t *ids, uint32_t n_ids)
1979 + struct pw_resource *resource = object;
1980 + struct resource_data *data = pw_resource_get_user_data(resource);
1983 + n_ids = SPA_MIN(n_ids, SPA_N_ELEMENTS(data->subscribe_ids));
1984 + data->n_subscribe_ids = n_ids;
1986 + for (i = 0; i < n_ids; i++) {
1987 + data->subscribe_ids[i] = ids[i];
1988 + pw_log_debug(NAME" %p: resource %d subscribe param %u",
1989 + data->link, resource->id, ids[i]);
1990 + endpoint_link_enum_params(resource, 1, ids[i], 0, UINT32_MAX, NULL);
1995 +static int endpoint_link_set_param (void *object, uint32_t id, uint32_t flags,
1996 + const struct spa_pod *param)
1998 + struct pw_resource *resource = object;
1999 + struct resource_data *data = pw_resource_get_user_data(resource);
2000 + struct endpoint_link *this = data->link;
2002 + pw_client_session_resource_set_param(this->client_sess->resource,
2003 + id, flags, param);
2008 +static int endpoint_link_request_state(void *object, enum pw_endpoint_link_state state)
2010 + struct pw_resource *resource = object;
2011 + struct resource_data *data = pw_resource_get_user_data(resource);
2012 + struct endpoint_link *this = data->link;
2014 + pw_client_session_resource_link_request_state(this->client_sess->resource,
2020 +static int endpoint_link_destroy(void *object)
2022 + struct pw_resource *resource = object;
2023 + struct resource_data *data = pw_resource_get_user_data(resource);
2024 + struct endpoint_link *this = data->link;
2026 + pw_client_session_resource_destroy_link(this->client_sess->resource,
2032 +static const struct pw_endpoint_link_proxy_methods methods = {
2033 + PW_VERSION_ENDPOINT_LINK_PROXY_METHODS,
2034 + .subscribe_params = endpoint_link_subscribe_params,
2035 + .enum_params = endpoint_link_enum_params,
2036 + .set_param = endpoint_link_set_param,
2037 + .request_state = endpoint_link_request_state,
2038 + .destroy = endpoint_link_destroy,
2041 +static void endpoint_link_notify_subscribed(struct endpoint_link *this,
2042 + uint32_t index, uint32_t next)
2044 + struct pw_global *global = this->global;
2045 + struct pw_resource *resource;
2046 + struct resource_data *data;
2047 + struct spa_pod *param = this->params[index];
2051 + if (!param || !spa_pod_is_object (param))
2054 + id = SPA_POD_OBJECT_ID (param);
2056 + spa_list_for_each(resource, &global->resource_list, link) {
2057 + data = pw_resource_get_user_data(resource);
2058 + for (i = 0; i < data->n_subscribe_ids; i++) {
2059 + if (data->subscribe_ids[i] == id) {
2060 + pw_endpoint_link_resource_param(resource, 1,
2061 + id, index, next, param);
2067 +int endpoint_link_update(struct endpoint_link *this,
2068 + uint32_t change_mask,
2069 + uint32_t n_params,
2070 + const struct spa_pod **params,
2071 + const struct pw_endpoint_link_info *info)
2073 + if (change_mask & PW_CLIENT_SESSION_UPDATE_PARAMS) {
2075 + size_t size = n_params * sizeof(struct spa_pod *);
2077 + pw_log_debug(NAME" %p: update %d params", this, n_params);
2079 + for (i = 0; i < this->n_params; i++)
2080 + free(this->params[i]);
2081 + this->params = realloc(this->params, size);
2082 + if (size > 0 && !this->params) {
2083 + this->n_params = 0;
2086 + this->n_params = n_params;
2088 + for (i = 0; i < this->n_params; i++) {
2089 + this->params[i] = params[i] ? spa_pod_copy(params[i]) : NULL;
2090 + endpoint_link_notify_subscribed(this, i, i+1);
2094 + if (change_mask & PW_CLIENT_SESSION_UPDATE_INFO) {
2095 + struct pw_resource *resource;
2097 + if (info->change_mask & PW_ENDPOINT_LINK_CHANGE_MASK_STATE) {
2098 + this->info.state = info->state;
2099 + free(this->info.error);
2100 + this->info.error = info->error ? strdup(info->error) : NULL;
2103 + if (info->change_mask & PW_ENDPOINT_LINK_CHANGE_MASK_PROPS)
2104 + pw_properties_update(this->props, info->props);
2106 + if (info->change_mask & PW_ENDPOINT_LINK_CHANGE_MASK_PARAMS) {
2107 + size_t size = info->n_params * sizeof(struct spa_param_info);
2109 + this->info.params = realloc(this->info.params, size);
2110 + if (size > 0 && !this->info.params) {
2111 + this->info.n_params = 0;
2114 + this->info.n_params = info->n_params;
2116 + memcpy(this->info.params, info->params, size);
2119 + if (!this->info.output_endpoint_id) {
2120 + this->info.output_endpoint_id = info->output_endpoint_id;
2121 + this->info.output_stream_id = info->output_stream_id;
2122 + this->info.input_endpoint_id = info->input_endpoint_id;
2123 + this->info.input_stream_id = info->input_stream_id;
2126 + this->info.change_mask = info->change_mask;
2127 + spa_list_for_each(resource, &this->global->resource_list, link) {
2128 + pw_endpoint_link_resource_info(resource, &this->info);
2130 + this->info.change_mask = 0;
2136 + pw_log_error(NAME" %p: can't update: no memory", this);
2137 + pw_resource_error(this->client_sess->resource, -ENOMEM,
2138 + NAME" %p: can't update: no memory", this);
2142 +static void endpoint_link_unbind(void *data)
2144 + struct pw_resource *resource = data;
2145 + spa_list_remove(&resource->link);
2148 +static const struct pw_resource_events resource_events = {
2149 + PW_VERSION_RESOURCE_EVENTS,
2150 + .destroy = endpoint_link_unbind,
2153 +static int endpoint_link_bind(void *_data, struct pw_client *client,
2154 + uint32_t permissions, uint32_t version, uint32_t id)
2156 + struct endpoint_link *this = _data;
2157 + struct pw_global *global = this->global;
2158 + struct pw_resource *resource;
2159 + struct resource_data *data;
2161 + resource = pw_resource_new(client, id, permissions, global->type, version, sizeof(*data));
2162 + if (resource == NULL)
2165 + data = pw_resource_get_user_data(resource);
2166 + data->link = this;
2167 + pw_resource_add_listener(resource, &data->resource_listener,
2168 + &resource_events, resource);
2169 + pw_resource_add_object_listener(resource, &data->object_listener,
2170 + &methods, resource);
2172 + pw_log_debug(NAME" %p: bound to %d", this, resource->id);
2174 + spa_list_append(&global->resource_list, &resource->link);
2176 + this->info.change_mask = PW_ENDPOINT_LINK_CHANGE_MASK_ALL;
2177 + pw_endpoint_link_resource_info(resource, &this->info);
2178 + this->info.change_mask = 0;
2183 + pw_log_error(NAME" %p: can't create resource: no memory", this);
2184 + pw_resource_error(this->client_sess->resource, -ENOMEM,
2185 + NAME" %p: can't create resource: no memory", this);
2189 +int endpoint_link_init(struct endpoint_link *this,
2190 + uint32_t id, uint32_t session_id,
2191 + struct client_session *client_sess,
2192 + struct pw_core *core,
2193 + struct pw_properties *properties)
2195 + pw_log_debug(NAME" %p: new", this);
2197 + this->client_sess = client_sess;
2199 + this->props = properties;
2201 + properties = pw_properties_copy(properties);
2205 + this->global = pw_global_new (core,
2206 + PW_TYPE_INTERFACE_EndpointLink,
2207 + PW_VERSION_ENDPOINT_LINK_PROXY,
2208 + properties, endpoint_link_bind, this);
2209 + if (!this->global)
2212 + this->info.version = PW_VERSION_ENDPOINT_LINK_INFO;
2213 + this->info.id = this->global->id;
2214 + this->info.session_id = session_id;
2215 + this->info.props = &this->props->dict;
2217 + return pw_global_register(this->global);
2220 + pw_log_error(NAME" - can't create - out of memory");
2224 +void endpoint_link_clear(struct endpoint_link *this)
2228 + pw_log_debug(NAME" %p: destroy", this);
2230 + pw_global_destroy(this->global);
2232 + for (i = 0; i < this->n_params; i++)
2233 + free(this->params[i]);
2234 + free(this->params);
2236 + free(this->info.error);
2237 + free(this->info.params);
2240 + pw_properties_free(this->props);
2242 diff --git a/src/modules/module-session-manager/endpoint-link.h b/src/modules/module-session-manager/endpoint-link.h
2243 new file mode 100644
2244 index 00000000..a9c18d32
2246 +++ b/src/modules/module-session-manager/endpoint-link.h
2250 + * Copyright © 2019 Collabora Ltd.
2251 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
2253 + * Permission is hereby granted, free of charge, to any person obtaining a
2254 + * copy of this software and associated documentation files (the "Software"),
2255 + * to deal in the Software without restriction, including without limitation
2256 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
2257 + * and/or sell copies of the Software, and to permit persons to whom the
2258 + * Software is furnished to do so, subject to the following conditions:
2260 + * The above copyright notice and this permission notice (including the next
2261 + * paragraph) shall be included in all copies or substantial portions of the
2264 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
2265 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
2266 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2267 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
2268 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
2269 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
2270 + * DEALINGS IN THE SOFTWARE.
2273 +#ifndef MODULE_SESSION_MANAGER_ENDPOINT_LINK_H
2274 +#define MODULE_SESSION_MANAGER_ENDPOINT_LINK_H
2280 +struct client_session;
2282 +struct endpoint_link {
2283 + struct spa_list link;
2284 + struct client_session *client_sess;
2285 + struct pw_global *global;
2286 + uint32_t id; /* session-local link id */
2287 + uint32_t n_params;
2288 + struct spa_pod **params;
2289 + struct pw_endpoint_link_info info;
2290 + struct pw_properties *props; /* wrapper of info.props */
2293 +int endpoint_link_init(struct endpoint_link *this,
2294 + uint32_t id, uint32_t session_id,
2295 + struct client_session *client_sess,
2296 + struct pw_core *core,
2297 + struct pw_properties *properties);
2299 +void endpoint_link_clear(struct endpoint_link *this);
2301 +int endpoint_link_update(struct endpoint_link *this,
2302 + uint32_t change_mask,
2303 + uint32_t n_params,
2304 + const struct spa_pod **params,
2305 + const struct pw_endpoint_link_info *info);
2311 +#endif /* MODULE_SESSION_MANAGER_ENDPOINT_LINK_H */
2312 diff --git a/src/modules/module-session-manager/endpoint-stream.c b/src/modules/module-session-manager/endpoint-stream.c
2313 new file mode 100644
2314 index 00000000..47d2a4ea
2316 +++ b/src/modules/module-session-manager/endpoint-stream.c
2320 + * Copyright © 2019 Collabora Ltd.
2321 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
2323 + * Permission is hereby granted, free of charge, to any person obtaining a
2324 + * copy of this software and associated documentation files (the "Software"),
2325 + * to deal in the Software without restriction, including without limitation
2326 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
2327 + * and/or sell copies of the Software, and to permit persons to whom the
2328 + * Software is furnished to do so, subject to the following conditions:
2330 + * The above copyright notice and this permission notice (including the next
2331 + * paragraph) shall be included in all copies or substantial portions of the
2334 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
2335 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
2336 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2337 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
2338 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
2339 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
2340 + * DEALINGS IN THE SOFTWARE.
2343 +#include <stdbool.h>
2344 +#include <string.h>
2346 +#include <pipewire/pipewire.h>
2347 +#include <extensions/session-manager.h>
2349 +#include <spa/pod/filter.h>
2351 +#include "endpoint-stream.h"
2352 +#include "client-endpoint.h"
2354 +#include <pipewire/private.h>
2356 +#define NAME "endpoint-stream"
2358 +struct resource_data {
2359 + struct endpoint_stream *stream;
2360 + struct spa_hook resource_listener;
2361 + struct spa_hook object_listener;
2362 + uint32_t n_subscribe_ids;
2363 + uint32_t subscribe_ids[32];
2366 +#define pw_endpoint_stream_resource(r,m,v,...) \
2367 + pw_resource_call(r,struct pw_endpoint_stream_proxy_events,m,v,__VA_ARGS__)
2368 +#define pw_endpoint_stream_resource_info(r,...) \
2369 + pw_endpoint_stream_resource(r,info,0,__VA_ARGS__)
2370 +#define pw_endpoint_stream_resource_param(r,...) \
2371 + pw_endpoint_stream_resource(r,param,0,__VA_ARGS__)
2373 +static int endpoint_stream_enum_params (void *object, int seq,
2374 + uint32_t id, uint32_t start, uint32_t num,
2375 + const struct spa_pod *filter)
2377 + struct pw_resource *resource = object;
2378 + struct resource_data *data = pw_resource_get_user_data(resource);
2379 + struct endpoint_stream *this = data->stream;
2380 + struct spa_pod *result;
2381 + struct spa_pod *param;
2382 + uint8_t buffer[1024];
2383 + struct spa_pod_builder b = { 0 };
2385 + uint32_t next = start;
2386 + uint32_t count = 0;
2390 + if (index >= this->n_params)
2393 + param = this->params[index];
2395 + if (param == NULL || !spa_pod_is_object_id(param, id))
2398 + spa_pod_builder_init(&b, buffer, sizeof(buffer));
2399 + if (spa_pod_filter(&b, &result, param, filter) != 0)
2402 + pw_log_debug(NAME" %p: %d param %u", this, seq, index);
2404 + pw_endpoint_stream_resource_param(resource, seq, id, index, next, result);
2406 + if (++count == num)
2412 +static int endpoint_stream_subscribe_params (void *object, uint32_t *ids, uint32_t n_ids)
2414 + struct pw_resource *resource = object;
2415 + struct resource_data *data = pw_resource_get_user_data(resource);
2418 + n_ids = SPA_MIN(n_ids, SPA_N_ELEMENTS(data->subscribe_ids));
2419 + data->n_subscribe_ids = n_ids;
2421 + for (i = 0; i < n_ids; i++) {
2422 + data->subscribe_ids[i] = ids[i];
2423 + pw_log_debug(NAME" %p: resource %d subscribe param %u",
2424 + data->stream, resource->id, ids[i]);
2425 + endpoint_stream_enum_params(resource, 1, ids[i], 0, UINT32_MAX, NULL);
2430 +static int endpoint_stream_set_param (void *object, uint32_t id, uint32_t flags,
2431 + const struct spa_pod *param)
2433 + struct pw_resource *resource = object;
2434 + struct resource_data *data = pw_resource_get_user_data(resource);
2435 + struct endpoint_stream *this = data->stream;
2437 + pw_client_endpoint_resource_set_param(this->client_ep->resource,
2438 + id, flags, param);
2443 +static const struct pw_endpoint_stream_proxy_methods methods = {
2444 + PW_VERSION_ENDPOINT_STREAM_PROXY_METHODS,
2445 + .subscribe_params = endpoint_stream_subscribe_params,
2446 + .enum_params = endpoint_stream_enum_params,
2447 + .set_param = endpoint_stream_set_param,
2450 +static void endpoint_stream_notify_subscribed(struct endpoint_stream *this,
2451 + uint32_t index, uint32_t next)
2453 + struct pw_global *global = this->global;
2454 + struct pw_resource *resource;
2455 + struct resource_data *data;
2456 + struct spa_pod *param = this->params[index];
2460 + if (!param || !spa_pod_is_object (param))
2463 + id = SPA_POD_OBJECT_ID (param);
2465 + spa_list_for_each(resource, &global->resource_list, link) {
2466 + data = pw_resource_get_user_data(resource);
2467 + for (i = 0; i < data->n_subscribe_ids; i++) {
2468 + if (data->subscribe_ids[i] == id) {
2469 + pw_endpoint_stream_resource_param(resource, 1,
2470 + id, index, next, param);
2476 +int endpoint_stream_update(struct endpoint_stream *this,
2477 + uint32_t change_mask,
2478 + uint32_t n_params,
2479 + const struct spa_pod **params,
2480 + const struct pw_endpoint_stream_info *info)
2482 + if (change_mask & PW_CLIENT_ENDPOINT_UPDATE_PARAMS) {
2484 + size_t size = n_params * sizeof(struct spa_pod *);
2486 + pw_log_debug(NAME" %p: update %d params", this, n_params);
2488 + for (i = 0; i < this->n_params; i++)
2489 + free(this->params[i]);
2490 + this->params = realloc(this->params, size);
2491 + if (size > 0 && !this->params) {
2492 + this->n_params = 0;
2495 + this->n_params = n_params;
2497 + for (i = 0; i < this->n_params; i++) {
2498 + this->params[i] = params[i] ? spa_pod_copy(params[i]) : NULL;
2499 + endpoint_stream_notify_subscribed(this, i, i+1);
2503 + if (change_mask & PW_CLIENT_ENDPOINT_UPDATE_INFO) {
2504 + struct pw_resource *resource;
2506 + if (info->change_mask & PW_ENDPOINT_STREAM_CHANGE_MASK_LINK_PARAMS) {
2507 + free(this->info.link_params);
2508 + this->info.link_params = spa_pod_copy(info->link_params);
2511 + if (info->change_mask & PW_ENDPOINT_STREAM_CHANGE_MASK_PROPS)
2512 + pw_properties_update(this->props, info->props);
2514 + if (info->change_mask & PW_ENDPOINT_STREAM_CHANGE_MASK_PARAMS) {
2515 + size_t size = info->n_params * sizeof(struct spa_param_info);
2517 + this->info.params = realloc(this->info.params, size);
2518 + if (size > 0 && !this->info.params) {
2519 + this->info.n_params = 0;
2522 + this->info.n_params = info->n_params;
2524 + memcpy(this->info.params, info->params, size);
2527 + if (!this->info.name)
2528 + this->info.name = strdup(info->name);
2530 + this->info.change_mask = info->change_mask;
2531 + spa_list_for_each(resource, &this->global->resource_list, link) {
2532 + pw_endpoint_stream_resource_info(resource, &this->info);
2534 + this->info.change_mask = 0;
2540 + pw_log_error(NAME" can't update: no memory");
2541 + pw_resource_error(this->client_ep->resource, -ENOMEM,
2542 + NAME" can't update: no memory");
2546 +static void endpoint_stream_unbind(void *data)
2548 + struct pw_resource *resource = data;
2549 + spa_list_remove(&resource->link);
2552 +static const struct pw_resource_events resource_events = {
2553 + PW_VERSION_RESOURCE_EVENTS,
2554 + .destroy = endpoint_stream_unbind,
2557 +static int endpoint_stream_bind(void *_data, struct pw_client *client,
2558 + uint32_t permissions, uint32_t version, uint32_t id)
2560 + struct endpoint_stream *this = _data;
2561 + struct pw_global *global = this->global;
2562 + struct pw_resource *resource;
2563 + struct resource_data *data;
2565 + resource = pw_resource_new(client, id, permissions, global->type, version, sizeof(*data));
2566 + if (resource == NULL)
2569 + data = pw_resource_get_user_data(resource);
2570 + data->stream = this;
2571 + pw_resource_add_listener(resource, &data->resource_listener,
2572 + &resource_events, resource);
2573 + pw_resource_add_object_listener(resource, &data->object_listener,
2574 + &methods, resource);
2576 + pw_log_debug(NAME" %p: bound to %d", this, resource->id);
2578 + spa_list_append(&global->resource_list, &resource->link);
2580 + this->info.change_mask = PW_ENDPOINT_STREAM_CHANGE_MASK_ALL;
2581 + pw_endpoint_stream_resource_info(resource, &this->info);
2582 + this->info.change_mask = 0;
2587 + pw_log_error(NAME" can't create resource: no memory");
2588 + pw_resource_error(this->client_ep->resource, -ENOMEM,
2589 + NAME" can't create resource: no memory");
2593 +int endpoint_stream_init(struct endpoint_stream *this,
2594 + uint32_t id, uint32_t endpoint_id,
2595 + struct client_endpoint *client_ep,
2596 + struct pw_core *core,
2597 + struct pw_properties *properties)
2599 + pw_log_debug(NAME" %p: new", this);
2601 + this->client_ep = client_ep;
2603 + this->props = properties;
2605 + properties = pw_properties_copy(properties);
2609 + this->global = pw_global_new (core,
2610 + PW_TYPE_INTERFACE_EndpointStream,
2611 + PW_VERSION_ENDPOINT_STREAM_PROXY,
2612 + properties, endpoint_stream_bind, this);
2613 + if (!this->global)
2616 + this->info.version = PW_VERSION_ENDPOINT_STREAM_INFO;
2617 + this->info.id = this->global->id;
2618 + this->info.endpoint_id = endpoint_id;
2619 + this->info.props = &this->props->dict;
2621 + return pw_global_register(this->global);
2624 + pw_log_error(NAME" - can't create - out of memory");
2628 +void endpoint_stream_clear(struct endpoint_stream *this)
2632 + pw_log_debug(NAME" %p: destroy", this);
2634 + pw_global_destroy(this->global);
2636 + for (i = 0; i < this->n_params; i++)
2637 + free(this->params[i]);
2638 + free(this->params);
2640 + free(this->info.name);
2641 + free(this->info.link_params);
2642 + free(this->info.params);
2645 + pw_properties_free(this->props);
2647 diff --git a/src/modules/module-session-manager/endpoint-stream.h b/src/modules/module-session-manager/endpoint-stream.h
2648 new file mode 100644
2649 index 00000000..99bd4836
2651 +++ b/src/modules/module-session-manager/endpoint-stream.h
2655 + * Copyright © 2019 Collabora Ltd.
2656 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
2658 + * Permission is hereby granted, free of charge, to any person obtaining a
2659 + * copy of this software and associated documentation files (the "Software"),
2660 + * to deal in the Software without restriction, including without limitation
2661 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
2662 + * and/or sell copies of the Software, and to permit persons to whom the
2663 + * Software is furnished to do so, subject to the following conditions:
2665 + * The above copyright notice and this permission notice (including the next
2666 + * paragraph) shall be included in all copies or substantial portions of the
2669 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
2670 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
2671 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2672 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
2673 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
2674 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
2675 + * DEALINGS IN THE SOFTWARE.
2678 +#ifndef MODULE_SESSION_MANAGER_ENDPOINT_STREAM_H
2679 +#define MODULE_SESSION_MANAGER_ENDPOINT_STREAM_H
2685 +struct client_endpoint;
2687 +struct endpoint_stream {
2688 + struct spa_list link;
2689 + struct client_endpoint *client_ep;
2690 + struct pw_global *global;
2691 + uint32_t id; /* endpoint-local stream id */
2692 + uint32_t n_params;
2693 + struct spa_pod **params;
2694 + struct pw_endpoint_stream_info info;
2695 + struct pw_properties *props; /* wrapper of info.props */
2698 +int endpoint_stream_init(struct endpoint_stream *this,
2699 + uint32_t id, uint32_t endpoint_id,
2700 + struct client_endpoint *client_ep,
2701 + struct pw_core *core,
2702 + struct pw_properties *properties);
2704 +void endpoint_stream_clear(struct endpoint_stream *this);
2706 +int endpoint_stream_update(struct endpoint_stream *this,
2707 + uint32_t change_mask,
2708 + uint32_t n_params,
2709 + const struct spa_pod **params,
2710 + const struct pw_endpoint_stream_info *info);
2716 +#endif /* MODULE_SESSION_MANAGER_ENDPOINT_STREAM_H */
2717 diff --git a/src/modules/module-session-manager/endpoint.c b/src/modules/module-session-manager/endpoint.c
2718 new file mode 100644
2719 index 00000000..0866e71d
2721 +++ b/src/modules/module-session-manager/endpoint.c
2725 + * Copyright © 2019 Collabora Ltd.
2726 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
2728 + * Permission is hereby granted, free of charge, to any person obtaining a
2729 + * copy of this software and associated documentation files (the "Software"),
2730 + * to deal in the Software without restriction, including without limitation
2731 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
2732 + * and/or sell copies of the Software, and to permit persons to whom the
2733 + * Software is furnished to do so, subject to the following conditions:
2735 + * The above copyright notice and this permission notice (including the next
2736 + * paragraph) shall be included in all copies or substantial portions of the
2739 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
2740 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
2741 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2742 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
2743 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
2744 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
2745 + * DEALINGS IN THE SOFTWARE.
2748 +#include <stdbool.h>
2749 +#include <string.h>
2751 +#include <pipewire/pipewire.h>
2752 +#include <extensions/session-manager.h>
2754 +#include <spa/pod/filter.h>
2756 +#include "endpoint.h"
2757 +#include "client-endpoint.h"
2759 +#include <pipewire/private.h>
2761 +#define NAME "endpoint"
2763 +struct resource_data {
2764 + struct endpoint *endpoint;
2765 + struct spa_hook resource_listener;
2766 + struct spa_hook object_listener;
2767 + uint32_t n_subscribe_ids;
2768 + uint32_t subscribe_ids[32];
2771 +#define pw_endpoint_resource(r,m,v,...) \
2772 + pw_resource_call(r,struct pw_endpoint_proxy_events,m,v,__VA_ARGS__)
2773 +#define pw_endpoint_resource_info(r,...) \
2774 + pw_endpoint_resource(r,info,0,__VA_ARGS__)
2775 +#define pw_endpoint_resource_param(r,...) \
2776 + pw_endpoint_resource(r,param,0,__VA_ARGS__)
2778 +static int endpoint_enum_params (void *object, int seq,
2779 + uint32_t id, uint32_t start, uint32_t num,
2780 + const struct spa_pod *filter)
2782 + struct pw_resource *resource = object;
2783 + struct resource_data *data = pw_resource_get_user_data(resource);
2784 + struct endpoint *this = data->endpoint;
2785 + struct spa_pod *result;
2786 + struct spa_pod *param;
2787 + uint8_t buffer[1024];
2788 + struct spa_pod_builder b = { 0 };
2790 + uint32_t next = start;
2791 + uint32_t count = 0;
2795 + if (index >= this->n_params)
2798 + param = this->params[index];
2800 + if (param == NULL || !spa_pod_is_object_id(param, id))
2803 + spa_pod_builder_init(&b, buffer, sizeof(buffer));
2804 + if (spa_pod_filter(&b, &result, param, filter) != 0)
2807 + pw_log_debug(NAME" %p: %d param %u", this, seq, index);
2809 + pw_endpoint_resource_param(resource, seq, id, index, next, result);
2811 + if (++count == num)
2817 +static int endpoint_subscribe_params (void *object, uint32_t *ids, uint32_t n_ids)
2819 + struct pw_resource *resource = object;
2820 + struct resource_data *data = pw_resource_get_user_data(resource);
2823 + n_ids = SPA_MIN(n_ids, SPA_N_ELEMENTS(data->subscribe_ids));
2824 + data->n_subscribe_ids = n_ids;
2826 + for (i = 0; i < n_ids; i++) {
2827 + data->subscribe_ids[i] = ids[i];
2828 + pw_log_debug(NAME" %p: resource %d subscribe param %u",
2829 + data->endpoint, resource->id, ids[i]);
2830 + endpoint_enum_params(resource, 1, ids[i], 0, UINT32_MAX, NULL);
2835 +static int endpoint_set_param (void *object, uint32_t id, uint32_t flags,
2836 + const struct spa_pod *param)
2838 + struct pw_resource *resource = object;
2839 + struct resource_data *data = pw_resource_get_user_data(resource);
2840 + struct endpoint *this = data->endpoint;
2842 + pw_client_endpoint_resource_set_param(this->client_ep->resource,
2843 + id, flags, param);
2848 +static const struct pw_endpoint_proxy_methods methods = {
2849 + PW_VERSION_ENDPOINT_PROXY_METHODS,
2850 + .subscribe_params = endpoint_subscribe_params,
2851 + .enum_params = endpoint_enum_params,
2852 + .set_param = endpoint_set_param,
2855 +static void endpoint_notify_subscribed(struct endpoint *this,
2856 + uint32_t index, uint32_t next)
2858 + struct pw_global *global = this->global;
2859 + struct pw_resource *resource;
2860 + struct resource_data *data;
2861 + struct spa_pod *param = this->params[index];
2865 + if (!param || !spa_pod_is_object (param))
2868 + id = SPA_POD_OBJECT_ID (param);
2870 + spa_list_for_each(resource, &global->resource_list, link) {
2871 + data = pw_resource_get_user_data(resource);
2872 + for (i = 0; i < data->n_subscribe_ids; i++) {
2873 + if (data->subscribe_ids[i] == id) {
2874 + pw_endpoint_resource_param(resource, 1, id,
2875 + index, next, param);
2881 +int endpoint_update(struct endpoint *this,
2882 + uint32_t change_mask,
2883 + uint32_t n_params,
2884 + const struct spa_pod **params,
2885 + const struct pw_endpoint_info *info)
2887 + if (change_mask & PW_CLIENT_ENDPOINT_UPDATE_PARAMS) {
2889 + size_t size = n_params * sizeof(struct spa_pod *);
2891 + pw_log_debug(NAME" %p: update %d params", this, n_params);
2893 + for (i = 0; i < this->n_params; i++)
2894 + free(this->params[i]);
2895 + this->params = realloc(this->params, size);
2896 + if (size > 0 && !this->params) {
2897 + this->n_params = 0;
2900 + this->n_params = n_params;
2902 + for (i = 0; i < this->n_params; i++) {
2903 + this->params[i] = params[i] ? spa_pod_copy(params[i]) : NULL;
2904 + endpoint_notify_subscribed(this, i, i+1);
2908 + if (change_mask & PW_CLIENT_ENDPOINT_UPDATE_INFO) {
2909 + struct pw_resource *resource;
2911 + if (info->change_mask & PW_ENDPOINT_CHANGE_MASK_STREAMS)
2912 + this->info.n_streams = info->n_streams;
2914 + if (info->change_mask & PW_ENDPOINT_CHANGE_MASK_SESSION)
2915 + this->info.session_id = info->session_id;
2917 + if (info->change_mask & PW_ENDPOINT_CHANGE_MASK_PROPS)
2918 + pw_properties_update(this->props, info->props);
2920 + if (info->change_mask & PW_ENDPOINT_CHANGE_MASK_PARAMS) {
2921 + size_t size = info->n_params * sizeof(struct spa_param_info);
2923 + this->info.params = realloc(this->info.params, size);
2924 + if (size > 0 && !this->info.params) {
2925 + this->info.n_params = 0;
2928 + this->info.n_params = info->n_params;
2930 + memcpy(this->info.params, info->params, size);
2933 + if (!this->info.name) {
2934 + this->info.name = strdup(info->name);
2935 + this->info.media_class = strdup(info->media_class);
2936 + this->info.direction = info->direction;
2937 + this->info.flags = info->flags;
2940 + this->info.change_mask = info->change_mask;
2941 + spa_list_for_each(resource, &this->global->resource_list, link) {
2942 + pw_endpoint_resource_info(resource, &this->info);
2944 + this->info.change_mask = 0;
2950 + pw_log_error(NAME" can't update: no memory");
2951 + pw_resource_error(this->client_ep->resource, -ENOMEM,
2952 + NAME" can't update: no memory");
2956 +static void endpoint_unbind(void *data)
2958 + struct pw_resource *resource = data;
2959 + spa_list_remove(&resource->link);
2962 +static const struct pw_resource_events resource_events = {
2963 + PW_VERSION_RESOURCE_EVENTS,
2964 + .destroy = endpoint_unbind,
2967 +static int endpoint_bind(void *_data, struct pw_client *client,
2968 + uint32_t permissions, uint32_t version, uint32_t id)
2970 + struct endpoint *this = _data;
2971 + struct pw_global *global = this->global;
2972 + struct pw_resource *resource;
2973 + struct resource_data *data;
2975 + resource = pw_resource_new(client, id, permissions, global->type, version, sizeof(*data));
2976 + if (resource == NULL)
2979 + data = pw_resource_get_user_data(resource);
2980 + data->endpoint = this;
2981 + pw_resource_add_listener(resource, &data->resource_listener,
2982 + &resource_events, resource);
2983 + pw_resource_add_object_listener(resource, &data->object_listener,
2984 + &methods, resource);
2986 + pw_log_debug(NAME" %p: bound to %d", this, resource->id);
2988 + spa_list_append(&global->resource_list, &resource->link);
2990 + this->info.change_mask = PW_ENDPOINT_CHANGE_MASK_ALL;
2991 + pw_endpoint_resource_info(resource, &this->info);
2992 + this->info.change_mask = 0;
2997 + pw_log_error(NAME" can't create resource: no memory");
2998 + pw_resource_error(this->client_ep->resource, -ENOMEM,
2999 + NAME" can't create resource: no memory");
3003 +int endpoint_init(struct endpoint *this,
3004 + struct client_endpoint *client_ep,
3005 + struct pw_core *core,
3006 + struct pw_properties *properties)
3008 + const char *keys[] = {
3009 + PW_KEY_FACTORY_ID,
3014 + pw_log_debug(NAME" %p: new", this);
3016 + this->client_ep = client_ep;
3017 + this->props = properties;
3019 + properties = pw_properties_new(NULL, NULL);
3023 + pw_properties_copy_keys(this->props, properties, keys);
3025 + this->global = pw_global_new (core,
3026 + PW_TYPE_INTERFACE_Endpoint,
3027 + PW_VERSION_ENDPOINT_PROXY,
3028 + properties, endpoint_bind, this);
3029 + if (!this->global)
3032 + pw_properties_setf(this->props, PW_KEY_ENDPOINT_ID, "%u", this->global->id);
3034 + this->info.version = PW_VERSION_ENDPOINT_INFO;
3035 + this->info.id = this->global->id;
3036 + this->info.props = &this->props->dict;
3038 + pw_client_endpoint_resource_set_id(client_ep->resource, this->global->id);
3040 + return pw_global_register(this->global);
3043 + pw_log_error(NAME" - can't create - out of memory");
3047 +void endpoint_clear(struct endpoint *this)
3051 + pw_log_debug(NAME" %p: destroy", this);
3053 + pw_global_destroy(this->global);
3055 + for (i = 0; i < this->n_params; i++)
3056 + free(this->params[i]);
3057 + free(this->params);
3059 + free(this->info.name);
3060 + free(this->info.media_class);
3061 + free(this->info.params);
3064 + pw_properties_free(this->props);
3066 diff --git a/src/modules/module-session-manager/endpoint.h b/src/modules/module-session-manager/endpoint.h
3067 new file mode 100644
3068 index 00000000..89d26028
3070 +++ b/src/modules/module-session-manager/endpoint.h
3074 + * Copyright © 2019 Collabora Ltd.
3075 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
3077 + * Permission is hereby granted, free of charge, to any person obtaining a
3078 + * copy of this software and associated documentation files (the "Software"),
3079 + * to deal in the Software without restriction, including without limitation
3080 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
3081 + * and/or sell copies of the Software, and to permit persons to whom the
3082 + * Software is furnished to do so, subject to the following conditions:
3084 + * The above copyright notice and this permission notice (including the next
3085 + * paragraph) shall be included in all copies or substantial portions of the
3088 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
3089 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
3090 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
3091 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
3092 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
3093 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
3094 + * DEALINGS IN THE SOFTWARE.
3097 +#ifndef MODULE_SESSION_MANAGER_ENDPOINT_H
3098 +#define MODULE_SESSION_MANAGER_ENDPOINT_H
3104 +struct client_endpoint;
3107 + struct client_endpoint *client_ep;
3108 + struct pw_global *global;
3109 + uint32_t n_params;
3110 + struct spa_pod **params;
3111 + struct pw_endpoint_info info;
3112 + struct pw_properties *props; /* wrapper of info.props */
3115 +int endpoint_init(struct endpoint *this,
3116 + struct client_endpoint *client_ep,
3117 + struct pw_core *core,
3118 + struct pw_properties *properties);
3120 +void endpoint_clear(struct endpoint *this);
3122 +int endpoint_update(struct endpoint *this,
3123 + uint32_t change_mask,
3124 + uint32_t n_params,
3125 + const struct spa_pod **params,
3126 + const struct pw_endpoint_info *info);
3132 +#endif /* MODULE_SESSION_MANAGER_ENDPOINT_H */
3133 diff --git a/src/modules/module-session-manager/protocol-native.c b/src/modules/module-session-manager/protocol-native.c
3134 new file mode 100644
3135 index 00000000..2c791ffc
3137 +++ b/src/modules/module-session-manager/protocol-native.c
3141 + * Copyright © 2019 Collabora Ltd.
3142 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
3144 + * Permission is hereby granted, free of charge, to any person obtaining a
3145 + * copy of this software and associated documentation files (the "Software"),
3146 + * to deal in the Software without restriction, including without limitation
3147 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
3148 + * and/or sell copies of the Software, and to permit persons to whom the
3149 + * Software is furnished to do so, subject to the following conditions:
3151 + * The above copyright notice and this permission notice (including the next
3152 + * paragraph) shall be included in all copies or substantial portions of the
3155 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
3156 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
3157 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
3158 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
3159 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
3160 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
3161 + * DEALINGS IN THE SOFTWARE.
3164 +#include <pipewire/pipewire.h>
3165 +#include <spa/pod/parser.h>
3167 +#include <extensions/session-manager.h>
3168 +#include <extensions/protocol-native.h>
3170 +static void push_dict(struct spa_pod_builder *b, const struct spa_dict *dict)
3172 + struct spa_pod_frame f;
3176 + n_items = dict ? dict->n_items : 0;
3178 + spa_pod_builder_push_struct(b, &f);
3179 + spa_pod_builder_add(b, SPA_POD_Int(n_items), NULL);
3180 + for (i = 0; i < n_items; i++) {
3181 + spa_pod_builder_add(b,
3182 + SPA_POD_String(dict->items[i].key),
3183 + SPA_POD_String(dict->items[i].value),
3186 + spa_pod_builder_pop(b, &f);
3189 +/* macro because of alloca() */
3190 +#define parse_dict(p, f, dict) \
3194 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3195 + spa_pod_parser_get(p, SPA_POD_Int(&(dict)->n_items), NULL) < 0) \
3198 + if ((dict)->n_items > 0) { \
3199 + (dict)->items = alloca((dict)->n_items * sizeof(struct spa_dict_item)); \
3200 + for (i = 0; i < (dict)->n_items; i++) { \
3201 + if (spa_pod_parser_get(p, \
3202 + SPA_POD_String(&(dict)->items[i].key), \
3203 + SPA_POD_String(&(dict)->items[i].value), \
3208 + spa_pod_parser_pop(p, f); \
3211 +static void push_param_infos(struct spa_pod_builder *b, uint32_t n_params,
3212 + const struct spa_param_info *params)
3214 + struct spa_pod_frame f;
3217 + spa_pod_builder_push_struct(b, &f);
3218 + spa_pod_builder_add(b, SPA_POD_Int(n_params), NULL);
3219 + for (i = 0; i < n_params; i++) {
3220 + spa_pod_builder_add(b,
3221 + SPA_POD_Id(params[i].id),
3222 + SPA_POD_Int(params[i].flags),
3225 + spa_pod_builder_pop(b, &f);
3228 +/* macro because of alloca() */
3229 +#define parse_param_infos(p, f, n_params_p, params_p) \
3233 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3234 + spa_pod_parser_get(p, SPA_POD_Int(n_params_p), NULL) < 0) \
3237 + if (*(n_params_p) > 0) { \
3238 + *(params_p) = alloca(*(n_params_p) * sizeof(struct spa_param_info)); \
3239 + for (i = 0; i < *(n_params_p); i++) { \
3240 + if (spa_pod_parser_get(p, \
3241 + SPA_POD_Id(&(*(params_p))[i].id), \
3242 + SPA_POD_Int(&(*(params_p))[i].flags), \
3247 + spa_pod_parser_pop(p, f); \
3250 +/***********************************************
3252 + ***********************************************/
3255 +marshal_pw_session_info(struct spa_pod_builder *b,
3256 + const struct pw_session_info *info)
3258 + struct spa_pod_frame f;
3260 + spa_pod_builder_push_struct(b, &f);
3261 + spa_pod_builder_add(b,
3262 + SPA_POD_Int(info->version),
3263 + SPA_POD_Int(info->id),
3264 + SPA_POD_Int(info->change_mask),
3266 + push_dict(b, info->props);
3267 + push_param_infos(b, info->n_params, info->params);
3268 + spa_pod_builder_pop(b, &f);
3271 +/* macro because of alloca() */
3272 +#define demarshal_pw_session_info(p, f, info) \
3274 + struct spa_pod_frame sub_f; \
3275 + uint32_t version; \
3277 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3278 + spa_pod_parser_get(p, \
3279 + SPA_POD_Int(&version), \
3280 + SPA_POD_Int(&(info)->id), \
3281 + SPA_POD_Int(&(info)->change_mask), \
3282 + SPA_POD_Int(&(info)->n_params), \
3283 + SPA_POD_Int(&(info)->props->n_items), \
3287 + (info)->change_mask &= PW_SESSION_CHANGE_MASK_ALL; \
3289 + parse_dict(p, &sub_f, (info)->props); \
3290 + parse_param_infos(p, &sub_f, &(info)->n_params, &(info)->params); \
3292 + spa_pod_parser_pop(p, f); \
3296 +marshal_pw_endpoint_info(struct spa_pod_builder *b,
3297 + const struct pw_endpoint_info *info)
3299 + struct spa_pod_frame f;
3301 + spa_pod_builder_push_struct(b, &f);
3302 + spa_pod_builder_add(b,
3303 + SPA_POD_Int(info->version),
3304 + SPA_POD_Int(info->id),
3305 + SPA_POD_String(info->name),
3306 + SPA_POD_String(info->media_class),
3307 + SPA_POD_Int(info->direction),
3308 + SPA_POD_Int(info->flags),
3309 + SPA_POD_Int(info->change_mask),
3311 + push_dict(b, info->props);
3312 + push_param_infos(b, info->n_params, info->params);
3313 + spa_pod_builder_pop(b, &f);
3316 +/* macro because of alloca() */
3317 +#define demarshal_pw_endpoint_info(p, f, info) \
3319 + struct spa_pod_frame sub_f; \
3320 + uint32_t version; \
3322 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3323 + spa_pod_parser_get(p, \
3324 + SPA_POD_Int(&version), \
3325 + SPA_POD_Int(&(info)->id), \
3326 + SPA_POD_String(&(info)->name), \
3327 + SPA_POD_String(&(info)->media_class), \
3328 + SPA_POD_Int(&(info)->direction), \
3329 + SPA_POD_Int(&(info)->flags), \
3330 + SPA_POD_Int(&(info)->change_mask), \
3334 + (info)->change_mask &= PW_ENDPOINT_CHANGE_MASK_ALL; \
3336 + parse_dict(p, &sub_f, (info)->props); \
3337 + parse_param_infos(p, &sub_f, &(info)->n_params, &(info)->params); \
3339 + spa_pod_parser_pop(p, f); \
3343 +marshal_pw_endpoint_stream_info(struct spa_pod_builder *b,
3344 + const struct pw_endpoint_stream_info *info)
3346 + struct spa_pod_frame f;
3348 + spa_pod_builder_push_struct(b, &f);
3349 + spa_pod_builder_add(b,
3350 + SPA_POD_Int(info->version),
3351 + SPA_POD_Int(info->id),
3352 + SPA_POD_Int(info->endpoint_id),
3353 + SPA_POD_String(info->name),
3354 + SPA_POD_Int(info->change_mask),
3355 + SPA_POD_Pod(info->link_params),
3357 + push_dict(b, info->props);
3358 + push_param_infos(b, info->n_params, info->params);
3359 + spa_pod_builder_pop(b, &f);
3362 +/* macro because of alloca() */
3363 +#define demarshal_pw_endpoint_stream_info(p, f, info) \
3365 + struct spa_pod_frame sub_f; \
3366 + uint32_t version; \
3368 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3369 + spa_pod_parser_get(p, \
3370 + SPA_POD_Int(&version), \
3371 + SPA_POD_Int(&(info)->id), \
3372 + SPA_POD_Int(&(info)->endpoint_id), \
3373 + SPA_POD_String(&(info)->name), \
3374 + SPA_POD_Int(&(info)->change_mask), \
3375 + SPA_POD_Pod(&(info)->link_params), \
3379 + (info)->change_mask &= PW_ENDPOINT_STREAM_CHANGE_MASK_ALL; \
3381 + parse_dict(p, &sub_f, (info)->props); \
3382 + parse_param_infos(p, &sub_f, &(info)->n_params, &(info)->params); \
3384 + spa_pod_parser_pop(p, f); \
3388 +marshal_pw_endpoint_link_info(struct spa_pod_builder *b,
3389 + const struct pw_endpoint_link_info *info)
3391 + struct spa_pod_frame f;
3393 + spa_pod_builder_push_struct(b, &f);
3394 + spa_pod_builder_add(b,
3395 + SPA_POD_Int(info->version),
3396 + SPA_POD_Int(info->id),
3397 + SPA_POD_Int(info->session_id),
3398 + SPA_POD_Int(info->output_endpoint_id),
3399 + SPA_POD_Int(info->output_stream_id),
3400 + SPA_POD_Int(info->input_endpoint_id),
3401 + SPA_POD_Int(info->input_stream_id),
3402 + SPA_POD_Int(info->change_mask),
3403 + SPA_POD_Int(info->state),
3404 + SPA_POD_String(info->error),
3406 + push_dict(b, info->props);
3407 + push_param_infos(b, info->n_params, info->params);
3408 + spa_pod_builder_pop(b, &f);
3411 +/* macro because of alloca() */
3412 +#define demarshal_pw_endpoint_link_info(p, f, info) \
3414 + struct spa_pod_frame sub_f; \
3415 + uint32_t version; \
3417 + if (spa_pod_parser_push_struct(p, f) < 0 || \
3418 + spa_pod_parser_get(p, \
3419 + SPA_POD_Int(&version), \
3420 + SPA_POD_Int(&(info)->id), \
3421 + SPA_POD_Int(&(info)->session_id), \
3422 + SPA_POD_Int(&(info)->output_endpoint_id), \
3423 + SPA_POD_Int(&(info)->output_stream_id), \
3424 + SPA_POD_Int(&(info)->input_endpoint_id), \
3425 + SPA_POD_Int(&(info)->input_stream_id), \
3426 + SPA_POD_Int(&(info)->change_mask), \
3427 + SPA_POD_Int(&(info)->state), \
3428 + SPA_POD_String(&(info)->error), \
3432 + (info)->change_mask &= PW_ENDPOINT_LINK_CHANGE_MASK_ALL; \
3434 + parse_dict(p, &sub_f, (info)->props); \
3435 + parse_param_infos(p, &sub_f, &(info)->n_params, &(info)->params); \
3437 + spa_pod_parser_pop(p, f); \
3440 +/***********************************************
3442 + ***********************************************/
3444 +static int client_endpoint_marshal_set_id (void *object, uint32_t id)
3446 + struct pw_resource *resource = object;
3447 + struct spa_pod_builder *b;
3449 + b = pw_protocol_native_begin_resource(resource,
3450 + PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_ID, NULL);
3452 + spa_pod_builder_add (b, SPA_POD_Int(id), NULL);
3454 + return pw_protocol_native_end_resource(resource, b);
3457 +static int client_endpoint_marshal_set_session_id (void *object, uint32_t id)
3459 + struct pw_resource *resource = object;
3460 + struct spa_pod_builder *b;
3462 + b = pw_protocol_native_begin_resource(resource,
3463 + PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_SESSION_ID, NULL);
3465 + spa_pod_builder_add (b, SPA_POD_Int(id), NULL);
3467 + return pw_protocol_native_end_resource(resource, b);
3470 +static int client_endpoint_marshal_set_param (void *object,
3471 + uint32_t id, uint32_t flags,
3472 + const struct spa_pod *param)
3474 + struct pw_resource *resource = object;
3475 + struct spa_pod_builder *b;
3477 + b = pw_protocol_native_begin_resource(resource,
3478 + PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_PARAM, NULL);
3480 + spa_pod_builder_add_struct(b,
3482 + SPA_POD_Int(flags),
3483 + SPA_POD_Pod(param));
3485 + return pw_protocol_native_end_resource(resource, b);
3488 +static int client_endpoint_marshal_stream_set_param (void *object,
3489 + uint32_t stream_id, uint32_t id,
3490 + uint32_t flags, const struct spa_pod *param)
3492 + struct pw_resource *resource = object;
3493 + struct spa_pod_builder *b;
3495 + b = pw_protocol_native_begin_resource(resource,
3496 + PW_CLIENT_ENDPOINT_PROXY_EVENT_STREAM_SET_PARAM, NULL);
3498 + spa_pod_builder_add_struct(b,
3499 + SPA_POD_Int(stream_id),
3501 + SPA_POD_Int(flags),
3502 + SPA_POD_Pod(param));
3504 + return pw_protocol_native_end_resource(resource, b);
3507 +static int client_endpoint_marshal_add_listener(void *object,
3508 + struct spa_hook *listener,
3509 + const struct pw_client_endpoint_proxy_events *events,
3512 + struct pw_proxy *proxy = object;
3513 + pw_proxy_add_object_listener(proxy, listener, events, data);
3517 +static int client_endpoint_marshal_update(void *object,
3518 + uint32_t change_mask,
3519 + uint32_t n_params,
3520 + const struct spa_pod **params,
3521 + const struct pw_endpoint_info *info)
3523 + struct pw_proxy *proxy = object;
3524 + struct spa_pod_builder *b;
3525 + struct spa_pod_frame f;
3528 + b = pw_protocol_native_begin_proxy(proxy,
3529 + PW_CLIENT_ENDPOINT_PROXY_METHOD_UPDATE, NULL);
3531 + spa_pod_builder_push_struct(b, &f);
3532 + spa_pod_builder_add(b,
3533 + SPA_POD_Int(change_mask),
3534 + SPA_POD_Int(n_params),
3537 + for (i = 0; i < n_params; i++)
3538 + spa_pod_builder_add(b, SPA_POD_Pod(params[i]), NULL);
3541 + marshal_pw_endpoint_info(b, info);
3543 + spa_pod_builder_add(b, SPA_POD_Pod(NULL), NULL);
3545 + spa_pod_builder_pop(b, &f);
3547 + return pw_protocol_native_end_proxy(proxy, b);
3550 +static int client_endpoint_marshal_stream_update(void *object,
3551 + uint32_t stream_id,
3552 + uint32_t change_mask,
3553 + uint32_t n_params,
3554 + const struct spa_pod **params,
3555 + const struct pw_endpoint_stream_info *info)
3557 + struct pw_proxy *proxy = object;
3558 + struct spa_pod_builder *b;
3559 + struct spa_pod_frame f;
3562 + b = pw_protocol_native_begin_proxy(proxy,
3563 + PW_CLIENT_ENDPOINT_PROXY_METHOD_STREAM_UPDATE, NULL);
3565 + spa_pod_builder_push_struct(b, &f);
3566 + spa_pod_builder_add(b,
3567 + SPA_POD_Int(stream_id),
3568 + SPA_POD_Int(change_mask),
3569 + SPA_POD_Int(n_params),
3572 + for (i = 0; i < n_params; i++)
3573 + spa_pod_builder_add(b, SPA_POD_Pod(params[i]), NULL);
3576 + marshal_pw_endpoint_stream_info(b, info);
3578 + spa_pod_builder_add(b, SPA_POD_Pod(NULL), NULL);
3580 + spa_pod_builder_pop(b, &f);
3582 + return pw_protocol_native_end_proxy(proxy, b);
3585 +static int client_endpoint_demarshal_set_id(void *object,
3586 + const struct pw_protocol_native_message *msg)
3588 + struct pw_proxy *proxy = object;
3589 + struct spa_pod_parser prs;
3592 + spa_pod_parser_init(&prs, msg->data, msg->size);
3593 + if (spa_pod_parser_get_struct(&prs,
3594 + SPA_POD_Int(&id)) < 0)
3597 + return pw_proxy_notify(proxy, struct pw_client_endpoint_proxy_events,
3601 +static int client_endpoint_demarshal_set_session_id(void *object,
3602 + const struct pw_protocol_native_message *msg)
3604 + struct pw_proxy *proxy = object;
3605 + struct spa_pod_parser prs;
3608 + spa_pod_parser_init(&prs, msg->data, msg->size);
3609 + if (spa_pod_parser_get_struct(&prs,
3610 + SPA_POD_Int(&id)) < 0)
3613 + return pw_proxy_notify(proxy, struct pw_client_endpoint_proxy_events,
3614 + set_session_id, 0, id);
3617 +static int client_endpoint_demarshal_set_param(void *object,
3618 + const struct pw_protocol_native_message *msg)
3620 + struct pw_proxy *proxy = object;
3621 + struct spa_pod_parser prs;
3622 + uint32_t id, flags;
3623 + const struct spa_pod *param = NULL;
3625 + spa_pod_parser_init(&prs, msg->data, msg->size);
3626 + if (spa_pod_parser_get_struct(&prs,
3628 + SPA_POD_Int(&flags),
3629 + SPA_POD_PodObject(¶m)) < 0)
3632 + return pw_proxy_notify(proxy, struct pw_client_endpoint_proxy_events,
3633 + set_param, 0, id, flags, param);
3636 +static int client_endpoint_demarshal_stream_set_param(void *object,
3637 + const struct pw_protocol_native_message *msg)
3639 + struct pw_proxy *proxy = object;
3640 + struct spa_pod_parser prs;
3641 + uint32_t stream_id, id, flags;
3642 + const struct spa_pod *param = NULL;
3644 + spa_pod_parser_init(&prs, msg->data, msg->size);
3645 + if (spa_pod_parser_get_struct(&prs,
3646 + SPA_POD_Int(&stream_id),
3648 + SPA_POD_Int(&flags),
3649 + SPA_POD_PodObject(¶m)) < 0)
3652 + return pw_proxy_notify(proxy, struct pw_client_endpoint_proxy_events,
3653 + stream_set_param, 0, stream_id, id, flags, param);
3656 +static int client_endpoint_demarshal_update(void *object,
3657 + const struct pw_protocol_native_message *msg)
3659 + struct pw_resource *resource = object;
3660 + struct spa_pod_parser prs[2];
3661 + struct spa_pod_frame f[2];
3662 + uint32_t change_mask, n_params;
3663 + const struct spa_pod **params = NULL;
3664 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
3665 + struct pw_endpoint_info info = { .props = &props }, *infop = NULL;
3666 + struct spa_pod *ipod;
3669 + spa_pod_parser_init(&prs[0], msg->data, msg->size);
3670 + if (spa_pod_parser_push_struct(&prs[0], &f[0]) < 0 ||
3671 + spa_pod_parser_get(&prs[0],
3672 + SPA_POD_Int(&change_mask),
3673 + SPA_POD_Int(&n_params), NULL) < 0)
3677 + params = alloca(n_params * sizeof(struct spa_pod *));
3678 + for (i = 0; i < n_params; i++)
3679 + if (spa_pod_parser_get(&prs[0],
3680 + SPA_POD_PodObject(¶ms[i]), NULL) < 0)
3683 + if (spa_pod_parser_get(&prs[0], SPA_POD_PodStruct(&ipod), NULL) < 0)
3687 + spa_pod_parser_pod(&prs[1], ipod);
3688 + demarshal_pw_endpoint_info(&prs[1], &f[1], infop);
3691 + return pw_resource_notify(resource, struct pw_client_endpoint_proxy_methods,
3692 + update, 0, change_mask, n_params, params, infop);
3695 +static int client_endpoint_demarshal_stream_update(void *object,
3696 + const struct pw_protocol_native_message *msg)
3698 + struct pw_resource *resource = object;
3699 + struct spa_pod_parser prs[2];
3700 + struct spa_pod_frame f[2];
3701 + uint32_t stream_id, change_mask, n_params;
3702 + const struct spa_pod **params = NULL;
3703 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
3704 + struct pw_endpoint_stream_info info = { .props = &props }, *infop = NULL;
3705 + struct spa_pod *ipod;
3708 + spa_pod_parser_init(&prs[0], msg->data, msg->size);
3709 + if (spa_pod_parser_push_struct(&prs[0], &f[0]) < 0 ||
3710 + spa_pod_parser_get(&prs[0],
3711 + SPA_POD_Int(&stream_id),
3712 + SPA_POD_Int(&change_mask),
3713 + SPA_POD_Int(&n_params), NULL) < 0)
3717 + params = alloca(n_params * sizeof(struct spa_pod *));
3718 + for (i = 0; i < n_params; i++)
3719 + if (spa_pod_parser_get(&prs[0],
3720 + SPA_POD_PodObject(¶ms[i]), NULL) < 0)
3723 + if (spa_pod_parser_get(&prs[0], SPA_POD_PodStruct(&ipod), NULL) < 0)
3727 + spa_pod_parser_pod(&prs[1], ipod);
3728 + demarshal_pw_endpoint_stream_info(&prs[1], &f[1], infop);
3731 + return pw_resource_notify(resource, struct pw_client_endpoint_proxy_methods,
3732 + stream_update, 0, stream_id, change_mask, n_params, params, infop);
3735 +static const struct pw_client_endpoint_proxy_events pw_protocol_native_client_endpoint_event_marshal = {
3736 + PW_VERSION_CLIENT_ENDPOINT_PROXY_EVENTS,
3737 + .set_id = client_endpoint_marshal_set_id,
3738 + .set_session_id = client_endpoint_marshal_set_session_id,
3739 + .set_param = client_endpoint_marshal_set_param,
3740 + .stream_set_param = client_endpoint_marshal_stream_set_param,
3743 +static const struct pw_protocol_native_demarshal
3744 +pw_protocol_native_client_endpoint_event_demarshal[PW_CLIENT_ENDPOINT_PROXY_EVENT_NUM] =
3746 + [PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_ID] = { client_endpoint_demarshal_set_id, 0 },
3747 + [PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_SESSION_ID] = { client_endpoint_demarshal_set_session_id, 0 },
3748 + [PW_CLIENT_ENDPOINT_PROXY_EVENT_SET_PARAM] = { client_endpoint_demarshal_set_param, 0 },
3749 + [PW_CLIENT_ENDPOINT_PROXY_EVENT_STREAM_SET_PARAM] = { client_endpoint_demarshal_stream_set_param, 0 },
3752 +static const struct pw_client_endpoint_proxy_methods pw_protocol_native_client_endpoint_method_marshal = {
3753 + PW_VERSION_CLIENT_ENDPOINT_PROXY_METHODS,
3754 + .add_listener = client_endpoint_marshal_add_listener,
3755 + .update = client_endpoint_marshal_update,
3756 + .stream_update = client_endpoint_marshal_stream_update,
3759 +static const struct pw_protocol_native_demarshal
3760 +pw_protocol_native_client_endpoint_method_demarshal[PW_CLIENT_ENDPOINT_PROXY_METHOD_NUM] =
3762 + [PW_CLIENT_ENDPOINT_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
3763 + [PW_CLIENT_ENDPOINT_PROXY_METHOD_UPDATE] = { client_endpoint_demarshal_update, 0 },
3764 + [PW_CLIENT_ENDPOINT_PROXY_METHOD_STREAM_UPDATE] = { client_endpoint_demarshal_stream_update, 0 },
3767 +static const struct pw_protocol_marshal pw_protocol_native_client_endpoint_marshal = {
3768 + PW_TYPE_INTERFACE_ClientEndpoint,
3769 + PW_VERSION_CLIENT_ENDPOINT_PROXY,
3770 + PW_CLIENT_ENDPOINT_PROXY_METHOD_NUM,
3771 + PW_CLIENT_ENDPOINT_PROXY_EVENT_NUM,
3772 + &pw_protocol_native_client_endpoint_method_marshal,
3773 + &pw_protocol_native_client_endpoint_method_demarshal,
3774 + &pw_protocol_native_client_endpoint_event_marshal,
3775 + &pw_protocol_native_client_endpoint_event_demarshal,
3778 +/***********************************************
3780 + ***********************************************/
3782 +static int client_session_marshal_set_id (void *object, uint32_t id)
3784 + struct pw_resource *resource = object;
3785 + struct spa_pod_builder *b;
3787 + b = pw_protocol_native_begin_resource(resource,
3788 + PW_CLIENT_SESSION_PROXY_EVENT_SET_ID, NULL);
3790 + spa_pod_builder_add (b, SPA_POD_Int(id), NULL);
3792 + return pw_protocol_native_end_resource(resource, b);
3795 +static int client_session_marshal_set_param (void *object,
3796 + uint32_t id, uint32_t flags,
3797 + const struct spa_pod *param)
3799 + struct pw_resource *resource = object;
3800 + struct spa_pod_builder *b;
3802 + b = pw_protocol_native_begin_resource(resource,
3803 + PW_CLIENT_SESSION_PROXY_EVENT_SET_PARAM, NULL);
3805 + spa_pod_builder_add_struct(b,
3807 + SPA_POD_Int(flags),
3808 + SPA_POD_Pod(param));
3810 + return pw_protocol_native_end_resource(resource, b);
3813 +static int client_session_marshal_link_set_param (void *object,
3814 + uint32_t link_id, uint32_t id,
3815 + uint32_t flags, const struct spa_pod *param)
3817 + struct pw_resource *resource = object;
3818 + struct spa_pod_builder *b;
3820 + b = pw_protocol_native_begin_resource(resource,
3821 + PW_CLIENT_SESSION_PROXY_EVENT_LINK_SET_PARAM, NULL);
3823 + spa_pod_builder_add_struct(b,
3824 + SPA_POD_Int(link_id),
3826 + SPA_POD_Int(flags),
3827 + SPA_POD_Pod(param));
3829 + return pw_protocol_native_end_resource(resource, b);
3832 +static int client_session_marshal_create_link(void *object,
3833 + const struct spa_dict *props)
3835 + struct pw_resource *resource = object;
3836 + struct spa_pod_builder *b;
3838 + spa_return_val_if_fail(props, -EINVAL);
3840 + b = pw_protocol_native_begin_resource(resource,
3841 + PW_CLIENT_SESSION_PROXY_EVENT_CREATE_LINK, NULL);
3843 + push_dict(b, props);
3845 + return pw_protocol_native_end_resource(resource, b);
3848 +static int client_session_marshal_destroy_link (void *object, uint32_t link_id)
3850 + struct pw_resource *resource = object;
3851 + struct spa_pod_builder *b;
3853 + b = pw_protocol_native_begin_resource(resource,
3854 + PW_CLIENT_SESSION_PROXY_EVENT_DESTROY_LINK, NULL);
3856 + spa_pod_builder_add(b, SPA_POD_Int(link_id), NULL);
3858 + return pw_protocol_native_end_resource(resource, b);
3861 +static int client_session_marshal_link_request_state (void *object,
3862 + uint32_t link_id, uint32_t state)
3864 + struct pw_resource *resource = object;
3865 + struct spa_pod_builder *b;
3867 + b = pw_protocol_native_begin_resource(resource,
3868 + PW_CLIENT_SESSION_PROXY_EVENT_LINK_REQUEST_STATE, NULL);
3870 + spa_pod_builder_add_struct(b,
3871 + SPA_POD_Int(link_id),
3872 + SPA_POD_Int(state));
3874 + return pw_protocol_native_end_resource(resource, b);
3877 +static int client_session_marshal_add_listener(void *object,
3878 + struct spa_hook *listener,
3879 + const struct pw_client_session_proxy_events *events,
3882 + struct pw_proxy *proxy = object;
3883 + pw_proxy_add_object_listener(proxy, listener, events, data);
3887 +static int client_session_marshal_update(void *object,
3888 + uint32_t change_mask,
3889 + uint32_t n_params,
3890 + const struct spa_pod **params,
3891 + const struct pw_session_info *info)
3893 + struct pw_proxy *proxy = object;
3894 + struct spa_pod_builder *b;
3895 + struct spa_pod_frame f;
3898 + b = pw_protocol_native_begin_proxy(proxy,
3899 + PW_CLIENT_SESSION_PROXY_METHOD_UPDATE, NULL);
3901 + spa_pod_builder_push_struct(b, &f);
3902 + spa_pod_builder_add(b,
3903 + SPA_POD_Int(change_mask),
3904 + SPA_POD_Int(n_params),
3907 + for (i = 0; i < n_params; i++)
3908 + spa_pod_builder_add(b, SPA_POD_Pod(params[i]), NULL);
3911 + marshal_pw_session_info(b, info);
3913 + spa_pod_builder_add(b, SPA_POD_Pod(NULL), NULL);
3915 + spa_pod_builder_pop(b, &f);
3917 + return pw_protocol_native_end_proxy(proxy, b);
3920 +static int client_session_marshal_link_update(void *object,
3922 + uint32_t change_mask,
3923 + uint32_t n_params,
3924 + const struct spa_pod **params,
3925 + const struct pw_endpoint_link_info *info)
3927 + struct pw_proxy *proxy = object;
3928 + struct spa_pod_builder *b;
3929 + struct spa_pod_frame f;
3932 + b = pw_protocol_native_begin_proxy(proxy,
3933 + PW_CLIENT_SESSION_PROXY_METHOD_LINK_UPDATE, NULL);
3935 + spa_pod_builder_push_struct(b, &f);
3936 + spa_pod_builder_add(b,
3937 + SPA_POD_Int(link_id),
3938 + SPA_POD_Int(change_mask),
3939 + SPA_POD_Int(n_params),
3942 + for (i = 0; i < n_params; i++)
3943 + spa_pod_builder_add(b, SPA_POD_Pod(params[i]), NULL);
3946 + marshal_pw_endpoint_link_info(b, info);
3948 + spa_pod_builder_add(b, SPA_POD_Pod(NULL), NULL);
3950 + spa_pod_builder_pop(b, &f);
3952 + return pw_protocol_native_end_proxy(proxy, b);
3955 +static int client_session_demarshal_set_id(void *object,
3956 + const struct pw_protocol_native_message *msg)
3958 + struct pw_proxy *proxy = object;
3959 + struct spa_pod_parser prs;
3962 + spa_pod_parser_init(&prs, msg->data, msg->size);
3963 + if (spa_pod_parser_get_struct(&prs,
3964 + SPA_POD_Int(&id)) < 0)
3967 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
3971 +static int client_session_demarshal_set_param(void *object,
3972 + const struct pw_protocol_native_message *msg)
3974 + struct pw_proxy *proxy = object;
3975 + struct spa_pod_parser prs;
3976 + uint32_t id, flags;
3977 + const struct spa_pod *param = NULL;
3979 + spa_pod_parser_init(&prs, msg->data, msg->size);
3980 + if (spa_pod_parser_get_struct(&prs,
3982 + SPA_POD_Int(&flags),
3983 + SPA_POD_PodObject(¶m)) < 0)
3986 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
3987 + set_param, 0, id, flags, param);
3990 +static int client_session_demarshal_link_set_param(void *object,
3991 + const struct pw_protocol_native_message *msg)
3993 + struct pw_proxy *proxy = object;
3994 + struct spa_pod_parser prs;
3995 + uint32_t link_id, id, flags;
3996 + const struct spa_pod *param = NULL;
3998 + spa_pod_parser_init(&prs, msg->data, msg->size);
3999 + if (spa_pod_parser_get_struct(&prs,
4000 + SPA_POD_Int(&link_id),
4002 + SPA_POD_Int(&flags),
4003 + SPA_POD_PodObject(¶m)) < 0)
4006 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
4007 + link_set_param, 0, link_id, id, flags, param);
4010 +static int client_session_demarshal_create_link(void *object,
4011 + const struct pw_protocol_native_message *msg)
4013 + struct pw_proxy *proxy = object;
4014 + struct spa_pod_parser prs;
4015 + struct spa_pod_frame f;
4016 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4018 + spa_pod_parser_init(&prs, msg->data, msg->size);
4019 + parse_dict(&prs, &f, &props);
4021 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
4022 + create_link, 0, &props);
4025 +static int client_session_demarshal_destroy_link(void *object,
4026 + const struct pw_protocol_native_message *msg)
4028 + struct pw_proxy *proxy = object;
4029 + struct spa_pod_parser prs;
4032 + spa_pod_parser_init(&prs, msg->data, msg->size);
4033 + if (spa_pod_parser_get_struct(&prs,
4034 + SPA_POD_Int(&link_id)) < 0)
4037 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
4038 + destroy_link, 0, link_id);
4041 +static int client_session_demarshal_link_request_state(void *object,
4042 + const struct pw_protocol_native_message *msg)
4044 + struct pw_proxy *proxy = object;
4045 + struct spa_pod_parser prs;
4046 + uint32_t link_id, state;
4048 + spa_pod_parser_init(&prs, msg->data, msg->size);
4049 + if (spa_pod_parser_get_struct(&prs,
4050 + SPA_POD_Int(&link_id),
4051 + SPA_POD_Int(&state)) < 0)
4054 + return pw_proxy_notify(proxy, struct pw_client_session_proxy_events,
4055 + link_request_state, 0, link_id, state);
4058 +static int client_session_demarshal_update(void *object,
4059 + const struct pw_protocol_native_message *msg)
4061 + struct pw_resource *resource = object;
4062 + struct spa_pod_parser prs[2];
4063 + struct spa_pod_frame f[2];
4064 + uint32_t change_mask, n_params;
4065 + const struct spa_pod **params = NULL;
4066 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4067 + struct pw_session_info info = { .props = &props }, *infop = NULL;
4068 + struct spa_pod *ipod;
4071 + spa_pod_parser_init(&prs[0], msg->data, msg->size);
4072 + if (spa_pod_parser_push_struct(&prs[0], &f[0]) < 0 ||
4073 + spa_pod_parser_get(&prs[0],
4074 + SPA_POD_Int(&change_mask),
4075 + SPA_POD_Int(&n_params), NULL) < 0)
4079 + params = alloca(n_params * sizeof(struct spa_pod *));
4080 + for (i = 0; i < n_params; i++)
4081 + if (spa_pod_parser_get(&prs[0],
4082 + SPA_POD_PodObject(¶ms[i]), NULL) < 0)
4085 + if (spa_pod_parser_get(&prs[0], SPA_POD_PodStruct(&ipod), NULL) < 0)
4089 + spa_pod_parser_pod(&prs[1], ipod);
4090 + demarshal_pw_session_info(&prs[1], &f[1], infop);
4093 + return pw_resource_notify(resource, struct pw_client_session_proxy_methods,
4094 + update, 0, change_mask, n_params, params, infop);
4097 +static int client_session_demarshal_link_update(void *object,
4098 + const struct pw_protocol_native_message *msg)
4100 + struct pw_resource *resource = object;
4101 + struct spa_pod_parser prs[2];
4102 + struct spa_pod_frame f[2];
4103 + uint32_t link_id, change_mask, n_params;
4104 + const struct spa_pod **params = NULL;
4105 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4106 + struct pw_endpoint_link_info info = { .props = &props }, *infop = NULL;
4107 + struct spa_pod *ipod;
4110 + spa_pod_parser_init(&prs[0], msg->data, msg->size);
4111 + if (spa_pod_parser_push_struct(&prs[0], &f[0]) < 0 ||
4112 + spa_pod_parser_get(&prs[0],
4113 + SPA_POD_Int(&link_id),
4114 + SPA_POD_Int(&change_mask),
4115 + SPA_POD_Int(&n_params), NULL) < 0)
4119 + params = alloca(n_params * sizeof(struct spa_pod *));
4120 + for (i = 0; i < n_params; i++)
4121 + if (spa_pod_parser_get(&prs[0],
4122 + SPA_POD_PodObject(¶ms[i]), NULL) < 0)
4125 + if (spa_pod_parser_get(&prs[0], SPA_POD_PodStruct(&ipod), NULL) < 0)
4129 + spa_pod_parser_pod(&prs[1], ipod);
4130 + demarshal_pw_endpoint_link_info(&prs[1], &f[1], infop);
4133 + return pw_resource_notify(resource, struct pw_client_session_proxy_methods,
4134 + link_update, 0, link_id, change_mask, n_params, params, infop);
4137 +static const struct pw_client_session_proxy_events pw_protocol_native_client_session_event_marshal = {
4138 + PW_VERSION_CLIENT_SESSION_PROXY_EVENTS,
4139 + .set_id = client_session_marshal_set_id,
4140 + .set_param = client_session_marshal_set_param,
4141 + .link_set_param = client_session_marshal_link_set_param,
4142 + .create_link = client_session_marshal_create_link,
4143 + .destroy_link = client_session_marshal_destroy_link,
4144 + .link_request_state = client_session_marshal_link_request_state,
4147 +static const struct pw_protocol_native_demarshal
4148 +pw_protocol_native_client_session_event_demarshal[PW_CLIENT_SESSION_PROXY_EVENT_NUM] =
4150 + [PW_CLIENT_SESSION_PROXY_EVENT_SET_ID] = { client_session_demarshal_set_id, 0 },
4151 + [PW_CLIENT_SESSION_PROXY_EVENT_SET_PARAM] = { client_session_demarshal_set_param, 0 },
4152 + [PW_CLIENT_SESSION_PROXY_EVENT_LINK_SET_PARAM] = { client_session_demarshal_link_set_param, 0 },
4153 + [PW_CLIENT_SESSION_PROXY_EVENT_CREATE_LINK] = { client_session_demarshal_create_link, 0 },
4154 + [PW_CLIENT_SESSION_PROXY_EVENT_DESTROY_LINK] = { client_session_demarshal_destroy_link, 0 },
4155 + [PW_CLIENT_SESSION_PROXY_EVENT_LINK_REQUEST_STATE] = { client_session_demarshal_link_request_state, 0 },
4158 +static const struct pw_client_session_proxy_methods pw_protocol_native_client_session_method_marshal = {
4159 + PW_VERSION_CLIENT_SESSION_PROXY_METHODS,
4160 + .add_listener = client_session_marshal_add_listener,
4161 + .update = client_session_marshal_update,
4162 + .link_update = client_session_marshal_link_update,
4165 +static const struct pw_protocol_native_demarshal
4166 +pw_protocol_native_client_session_method_demarshal[PW_CLIENT_SESSION_PROXY_METHOD_NUM] =
4168 + [PW_CLIENT_SESSION_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
4169 + [PW_CLIENT_SESSION_PROXY_METHOD_UPDATE] = { client_session_demarshal_update, 0 },
4170 + [PW_CLIENT_SESSION_PROXY_METHOD_LINK_UPDATE] = { client_session_demarshal_link_update, 0 },
4173 +static const struct pw_protocol_marshal pw_protocol_native_client_session_marshal = {
4174 + PW_TYPE_INTERFACE_ClientSession,
4175 + PW_VERSION_CLIENT_SESSION_PROXY,
4176 + PW_CLIENT_SESSION_PROXY_METHOD_NUM,
4177 + PW_CLIENT_SESSION_PROXY_EVENT_NUM,
4178 + &pw_protocol_native_client_session_method_marshal,
4179 + &pw_protocol_native_client_session_method_demarshal,
4180 + &pw_protocol_native_client_session_event_marshal,
4181 + &pw_protocol_native_client_session_event_demarshal,
4184 +/***********************************************
4186 + ***********************************************/
4188 +static void endpoint_link_marshal_info (void *object,
4189 + const struct pw_endpoint_link_info *info)
4191 + struct pw_resource *resource = object;
4192 + struct spa_pod_builder *b;
4194 + b = pw_protocol_native_begin_resource(resource,
4195 + PW_ENDPOINT_LINK_PROXY_EVENT_INFO, NULL);
4197 + marshal_pw_endpoint_link_info(b, info);
4199 + pw_protocol_native_end_resource(resource, b);
4202 +static void endpoint_link_marshal_param (void *object, int seq, uint32_t id,
4203 + uint32_t index, uint32_t next,
4204 + const struct spa_pod *param)
4206 + struct pw_resource *resource = object;
4207 + struct spa_pod_builder *b;
4209 + b = pw_protocol_native_begin_resource(resource,
4210 + PW_ENDPOINT_LINK_PROXY_EVENT_PARAM, NULL);
4212 + spa_pod_builder_add_struct(b,
4215 + SPA_POD_Int(index),
4216 + SPA_POD_Int(next),
4217 + SPA_POD_Pod(param));
4219 + pw_protocol_native_end_resource(resource, b);
4222 +static int endpoint_link_marshal_add_listener(void *object,
4223 + struct spa_hook *listener,
4224 + const struct pw_endpoint_link_proxy_events *events,
4227 + struct pw_proxy *proxy = object;
4228 + pw_proxy_add_object_listener(proxy, listener, events, data);
4232 +static int endpoint_link_marshal_subscribe_params(void *object,
4233 + uint32_t *ids, uint32_t n_ids)
4235 + struct pw_proxy *proxy = object;
4236 + struct spa_pod_builder *b;
4238 + b = pw_protocol_native_begin_proxy(proxy,
4239 + PW_ENDPOINT_LINK_PROXY_METHOD_SUBSCRIBE_PARAMS, NULL);
4241 + spa_pod_builder_add_struct(b,
4242 + SPA_POD_Array(sizeof(uint32_t), SPA_TYPE_Id, n_ids, ids));
4244 + return pw_protocol_native_end_proxy(proxy, b);
4247 +static int endpoint_link_marshal_enum_params(void *object,
4248 + int seq, uint32_t id,
4249 + uint32_t start, uint32_t num,
4250 + const struct spa_pod *filter)
4252 + struct pw_protocol_native_message *msg;
4253 + struct pw_proxy *proxy = object;
4254 + struct spa_pod_builder *b;
4256 + b = pw_protocol_native_begin_proxy(proxy,
4257 + PW_ENDPOINT_LINK_PROXY_METHOD_ENUM_PARAMS, &msg);
4259 + spa_pod_builder_add_struct(b,
4260 + SPA_POD_Int(SPA_RESULT_RETURN_ASYNC(msg->seq)),
4262 + SPA_POD_Int(index),
4264 + SPA_POD_Pod(filter));
4266 + return pw_protocol_native_end_proxy(proxy, b);
4269 +static int endpoint_link_marshal_set_param(void *object,
4270 + uint32_t id, uint32_t flags,
4271 + const struct spa_pod *param)
4273 + struct pw_proxy *proxy = object;
4274 + struct spa_pod_builder *b;
4276 + b = pw_protocol_native_begin_proxy(proxy,
4277 + PW_ENDPOINT_LINK_PROXY_METHOD_SET_PARAM, NULL);
4279 + spa_pod_builder_add_struct(b,
4281 + SPA_POD_Int(flags),
4282 + SPA_POD_Pod(param));
4284 + return pw_protocol_native_end_proxy(proxy, b);
4287 +static int endpoint_link_marshal_request_state(void *object,
4288 + enum pw_endpoint_link_state state)
4290 + struct pw_proxy *proxy = object;
4291 + struct spa_pod_builder *b;
4293 + b = pw_protocol_native_begin_proxy(proxy,
4294 + PW_ENDPOINT_LINK_PROXY_METHOD_REQUEST_STATE, NULL);
4296 + spa_pod_builder_add_struct(b, SPA_POD_Int(state));
4298 + return pw_protocol_native_end_proxy(proxy, b);
4301 +static int endpoint_link_marshal_destroy(void *object)
4303 + struct pw_proxy *proxy = object;
4304 + struct spa_pod_builder *b;
4306 + b = pw_protocol_native_begin_proxy(proxy,
4307 + PW_ENDPOINT_LINK_PROXY_METHOD_DESTROY, NULL);
4309 + return pw_protocol_native_end_proxy(proxy, b);
4312 +static int endpoint_link_demarshal_info(void *object,
4313 + const struct pw_protocol_native_message *msg)
4315 + struct pw_proxy *proxy = object;
4316 + struct spa_pod_parser prs;
4317 + struct spa_pod_frame f;
4318 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4319 + struct pw_endpoint_link_info info = { .props = &props };
4321 + spa_pod_parser_init(&prs, msg->data, msg->size);
4323 + demarshal_pw_endpoint_link_info(&prs, &f, &info);
4325 + return pw_proxy_notify(proxy, struct pw_endpoint_link_proxy_events,
4329 +static int endpoint_link_demarshal_param(void *object,
4330 + const struct pw_protocol_native_message *msg)
4332 + struct pw_proxy *proxy = object;
4333 + struct spa_pod_parser prs;
4334 + uint32_t id, index, next;
4336 + struct spa_pod *param;
4338 + spa_pod_parser_init(&prs, msg->data, msg->size);
4339 + if (spa_pod_parser_get_struct(&prs,
4340 + SPA_POD_Int(&seq),
4342 + SPA_POD_Int(&index),
4343 + SPA_POD_Int(&next),
4344 + SPA_POD_Pod(¶m)) < 0)
4347 + return pw_proxy_notify(proxy, struct pw_endpoint_link_proxy_events,
4348 + param, 0, seq, id, index, next, param);
4351 +static int endpoint_link_demarshal_subscribe_params(void *object,
4352 + const struct pw_protocol_native_message *msg)
4354 + struct pw_resource *resource = object;
4355 + struct spa_pod_parser prs;
4356 + uint32_t csize, ctype, n_ids;
4359 + spa_pod_parser_init(&prs, msg->data, msg->size);
4360 + if (spa_pod_parser_get_struct(&prs,
4361 + SPA_POD_Array(&csize, &ctype, &n_ids, &ids)) < 0)
4364 + if (ctype != SPA_TYPE_Id)
4367 + return pw_resource_notify(resource, struct pw_endpoint_link_proxy_methods,
4368 + subscribe_params, 0, ids, n_ids);
4371 +static int endpoint_link_demarshal_enum_params(void *object,
4372 + const struct pw_protocol_native_message *msg)
4374 + struct pw_resource *resource = object;
4375 + struct spa_pod_parser prs;
4376 + uint32_t id, index, num;
4378 + struct spa_pod *filter;
4380 + spa_pod_parser_init(&prs, msg->data, msg->size);
4381 + if (spa_pod_parser_get_struct(&prs,
4382 + SPA_POD_Int(&seq),
4384 + SPA_POD_Int(&index),
4385 + SPA_POD_Int(&num),
4386 + SPA_POD_Pod(&filter)) < 0)
4389 + return pw_resource_notify(resource, struct pw_endpoint_link_proxy_methods,
4390 + enum_params, 0, seq, id, index, num, filter);
4393 +static int endpoint_link_demarshal_set_param(void *object,
4394 + const struct pw_protocol_native_message *msg)
4396 + struct pw_resource *resource = object;
4397 + struct spa_pod_parser prs;
4398 + uint32_t id, flags;
4399 + struct spa_pod *param;
4401 + spa_pod_parser_init(&prs, msg->data, msg->size);
4402 + if (spa_pod_parser_get_struct(&prs,
4404 + SPA_POD_Int(&flags),
4405 + SPA_POD_Pod(¶m)) < 0)
4408 + return pw_resource_notify(resource, struct pw_endpoint_link_proxy_methods,
4409 + set_param, 0, id, flags, param);
4412 +static int endpoint_link_demarshal_request_state(void *object,
4413 + const struct pw_protocol_native_message *msg)
4415 + struct pw_resource *resource = object;
4416 + struct spa_pod_parser prs;
4417 + enum pw_endpoint_link_state state;
4419 + spa_pod_parser_init(&prs, msg->data, msg->size);
4420 + if (spa_pod_parser_get_struct(&prs,
4421 + SPA_POD_Int(&state)) < 0)
4424 + return pw_resource_notify(resource, struct pw_endpoint_link_proxy_methods,
4425 + request_state, 0, state);
4428 +static int endpoint_link_demarshal_destroy(void *object,
4429 + const struct pw_protocol_native_message *msg)
4431 + struct pw_resource *resource = object;
4433 + return pw_resource_notify(resource, struct pw_endpoint_link_proxy_methods,
4437 +static const struct pw_endpoint_link_proxy_events pw_protocol_native_endpoint_link_event_marshal = {
4438 + PW_VERSION_ENDPOINT_LINK_PROXY_EVENTS,
4439 + .info = endpoint_link_marshal_info,
4440 + .param = endpoint_link_marshal_param,
4443 +static const struct pw_protocol_native_demarshal
4444 +pw_protocol_native_endpoint_link_event_demarshal[PW_ENDPOINT_LINK_PROXY_EVENT_NUM] =
4446 + [PW_ENDPOINT_LINK_PROXY_EVENT_INFO] = { endpoint_link_demarshal_info, 0 },
4447 + [PW_ENDPOINT_LINK_PROXY_EVENT_PARAM] = { endpoint_link_demarshal_param, 0 },
4450 +static const struct pw_endpoint_link_proxy_methods pw_protocol_native_endpoint_link_method_marshal = {
4451 + PW_VERSION_ENDPOINT_LINK_PROXY_METHODS,
4452 + .add_listener = endpoint_link_marshal_add_listener,
4453 + .subscribe_params = endpoint_link_marshal_subscribe_params,
4454 + .enum_params = endpoint_link_marshal_enum_params,
4455 + .set_param = endpoint_link_marshal_set_param,
4456 + .request_state = endpoint_link_marshal_request_state,
4457 + .destroy = endpoint_link_marshal_destroy,
4460 +static const struct pw_protocol_native_demarshal
4461 +pw_protocol_native_endpoint_link_method_demarshal[PW_ENDPOINT_LINK_PROXY_METHOD_NUM] =
4463 + [PW_ENDPOINT_LINK_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
4464 + [PW_ENDPOINT_LINK_PROXY_METHOD_SUBSCRIBE_PARAMS] = { endpoint_link_demarshal_subscribe_params, 0 },
4465 + [PW_ENDPOINT_LINK_PROXY_METHOD_ENUM_PARAMS] = { endpoint_link_demarshal_enum_params, 0 },
4466 + [PW_ENDPOINT_LINK_PROXY_METHOD_SET_PARAM] = { endpoint_link_demarshal_set_param, PW_PERM_W },
4467 + [PW_ENDPOINT_LINK_PROXY_METHOD_REQUEST_STATE] = { endpoint_link_demarshal_request_state, PW_PERM_W },
4468 + [PW_ENDPOINT_LINK_PROXY_METHOD_DESTROY] = { endpoint_link_demarshal_destroy, PW_PERM_W },
4471 +static const struct pw_protocol_marshal pw_protocol_native_endpoint_link_marshal = {
4472 + PW_TYPE_INTERFACE_EndpointLink,
4473 + PW_VERSION_ENDPOINT_LINK_PROXY,
4474 + PW_ENDPOINT_LINK_PROXY_METHOD_NUM,
4475 + PW_ENDPOINT_LINK_PROXY_EVENT_NUM,
4476 + &pw_protocol_native_endpoint_link_method_marshal,
4477 + &pw_protocol_native_endpoint_link_method_demarshal,
4478 + &pw_protocol_native_endpoint_link_event_marshal,
4479 + &pw_protocol_native_endpoint_link_event_demarshal,
4482 +/***********************************************
4484 + ***********************************************/
4486 +static void endpoint_stream_marshal_info (void *object,
4487 + const struct pw_endpoint_stream_info *info)
4489 + struct pw_resource *resource = object;
4490 + struct spa_pod_builder *b;
4492 + b = pw_protocol_native_begin_resource(resource,
4493 + PW_ENDPOINT_STREAM_PROXY_EVENT_INFO, NULL);
4495 + marshal_pw_endpoint_stream_info(b, info);
4497 + pw_protocol_native_end_resource(resource, b);
4500 +static void endpoint_stream_marshal_param (void *object, int seq, uint32_t id,
4501 + uint32_t index, uint32_t next,
4502 + const struct spa_pod *param)
4504 + struct pw_resource *resource = object;
4505 + struct spa_pod_builder *b;
4507 + b = pw_protocol_native_begin_resource(resource,
4508 + PW_ENDPOINT_STREAM_PROXY_EVENT_PARAM, NULL);
4510 + spa_pod_builder_add_struct(b,
4513 + SPA_POD_Int(index),
4514 + SPA_POD_Int(next),
4515 + SPA_POD_Pod(param));
4517 + pw_protocol_native_end_resource(resource, b);
4520 +static int endpoint_stream_marshal_add_listener(void *object,
4521 + struct spa_hook *listener,
4522 + const struct pw_endpoint_stream_proxy_events *events,
4525 + struct pw_proxy *proxy = object;
4526 + pw_proxy_add_object_listener(proxy, listener, events, data);
4530 +static int endpoint_stream_marshal_subscribe_params(void *object,
4531 + uint32_t *ids, uint32_t n_ids)
4533 + struct pw_proxy *proxy = object;
4534 + struct spa_pod_builder *b;
4536 + b = pw_protocol_native_begin_proxy(proxy,
4537 + PW_ENDPOINT_STREAM_PROXY_METHOD_SUBSCRIBE_PARAMS, NULL);
4539 + spa_pod_builder_add_struct(b,
4540 + SPA_POD_Array(sizeof(uint32_t), SPA_TYPE_Id, n_ids, ids));
4542 + return pw_protocol_native_end_proxy(proxy, b);
4545 +static int endpoint_stream_marshal_enum_params(void *object,
4546 + int seq, uint32_t id,
4547 + uint32_t start, uint32_t num,
4548 + const struct spa_pod *filter)
4550 + struct pw_protocol_native_message *msg;
4551 + struct pw_proxy *proxy = object;
4552 + struct spa_pod_builder *b;
4554 + b = pw_protocol_native_begin_proxy(proxy,
4555 + PW_ENDPOINT_STREAM_PROXY_METHOD_ENUM_PARAMS, &msg);
4557 + spa_pod_builder_add_struct(b,
4558 + SPA_POD_Int(SPA_RESULT_RETURN_ASYNC(msg->seq)),
4560 + SPA_POD_Int(index),
4562 + SPA_POD_Pod(filter));
4564 + return pw_protocol_native_end_proxy(proxy, b);
4567 +static int endpoint_stream_marshal_set_param(void *object,
4568 + uint32_t id, uint32_t flags,
4569 + const struct spa_pod *param)
4571 + struct pw_proxy *proxy = object;
4572 + struct spa_pod_builder *b;
4574 + b = pw_protocol_native_begin_proxy(proxy,
4575 + PW_ENDPOINT_STREAM_PROXY_METHOD_SET_PARAM, NULL);
4577 + spa_pod_builder_add_struct(b,
4579 + SPA_POD_Int(flags),
4580 + SPA_POD_Pod(param));
4582 + return pw_protocol_native_end_proxy(proxy, b);
4585 +static int endpoint_stream_demarshal_info(void *object,
4586 + const struct pw_protocol_native_message *msg)
4588 + struct pw_proxy *proxy = object;
4589 + struct spa_pod_parser prs;
4590 + struct spa_pod_frame f;
4591 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4592 + struct pw_endpoint_stream_info info = { .props = &props };
4594 + spa_pod_parser_init(&prs, msg->data, msg->size);
4596 + demarshal_pw_endpoint_stream_info(&prs, &f, &info);
4598 + return pw_proxy_notify(proxy, struct pw_endpoint_stream_proxy_events,
4602 +static int endpoint_stream_demarshal_param(void *object,
4603 + const struct pw_protocol_native_message *msg)
4605 + struct pw_proxy *proxy = object;
4606 + struct spa_pod_parser prs;
4607 + uint32_t id, index, next;
4609 + struct spa_pod *param;
4611 + spa_pod_parser_init(&prs, msg->data, msg->size);
4612 + if (spa_pod_parser_get_struct(&prs,
4613 + SPA_POD_Int(&seq),
4615 + SPA_POD_Int(&index),
4616 + SPA_POD_Int(&next),
4617 + SPA_POD_Pod(¶m)) < 0)
4620 + return pw_proxy_notify(proxy, struct pw_endpoint_stream_proxy_events,
4621 + param, 0, seq, id, index, next, param);
4624 +static int endpoint_stream_demarshal_subscribe_params(void *object,
4625 + const struct pw_protocol_native_message *msg)
4627 + struct pw_resource *resource = object;
4628 + struct spa_pod_parser prs;
4629 + uint32_t csize, ctype, n_ids;
4632 + spa_pod_parser_init(&prs, msg->data, msg->size);
4633 + if (spa_pod_parser_get_struct(&prs,
4634 + SPA_POD_Array(&csize, &ctype, &n_ids, &ids)) < 0)
4637 + if (ctype != SPA_TYPE_Id)
4640 + return pw_resource_notify(resource, struct pw_endpoint_stream_proxy_methods,
4641 + subscribe_params, 0, ids, n_ids);
4644 +static int endpoint_stream_demarshal_enum_params(void *object,
4645 + const struct pw_protocol_native_message *msg)
4647 + struct pw_resource *resource = object;
4648 + struct spa_pod_parser prs;
4649 + uint32_t id, index, num;
4651 + struct spa_pod *filter;
4653 + spa_pod_parser_init(&prs, msg->data, msg->size);
4654 + if (spa_pod_parser_get_struct(&prs,
4655 + SPA_POD_Int(&seq),
4657 + SPA_POD_Int(&index),
4658 + SPA_POD_Int(&num),
4659 + SPA_POD_Pod(&filter)) < 0)
4662 + return pw_resource_notify(resource, struct pw_endpoint_stream_proxy_methods,
4663 + enum_params, 0, seq, id, index, num, filter);
4666 +static int endpoint_stream_demarshal_set_param(void *object,
4667 + const struct pw_protocol_native_message *msg)
4669 + struct pw_resource *resource = object;
4670 + struct spa_pod_parser prs;
4671 + uint32_t id, flags;
4672 + struct spa_pod *param;
4674 + spa_pod_parser_init(&prs, msg->data, msg->size);
4675 + if (spa_pod_parser_get_struct(&prs,
4677 + SPA_POD_Int(&flags),
4678 + SPA_POD_Pod(¶m)) < 0)
4681 + return pw_resource_notify(resource, struct pw_endpoint_stream_proxy_methods,
4682 + set_param, 0, id, flags, param);
4685 +static const struct pw_endpoint_stream_proxy_events pw_protocol_native_endpoint_stream_event_marshal = {
4686 + PW_VERSION_ENDPOINT_STREAM_PROXY_EVENTS,
4687 + .info = endpoint_stream_marshal_info,
4688 + .param = endpoint_stream_marshal_param,
4691 +static const struct pw_protocol_native_demarshal
4692 +pw_protocol_native_endpoint_stream_event_demarshal[PW_ENDPOINT_STREAM_PROXY_EVENT_NUM] =
4694 + [PW_ENDPOINT_STREAM_PROXY_EVENT_INFO] = { endpoint_stream_demarshal_info, 0 },
4695 + [PW_ENDPOINT_STREAM_PROXY_EVENT_PARAM] = { endpoint_stream_demarshal_param, 0 },
4698 +static const struct pw_endpoint_stream_proxy_methods pw_protocol_native_endpoint_stream_method_marshal = {
4699 + PW_VERSION_ENDPOINT_STREAM_PROXY_METHODS,
4700 + .add_listener = endpoint_stream_marshal_add_listener,
4701 + .subscribe_params = endpoint_stream_marshal_subscribe_params,
4702 + .enum_params = endpoint_stream_marshal_enum_params,
4703 + .set_param = endpoint_stream_marshal_set_param,
4706 +static const struct pw_protocol_native_demarshal
4707 +pw_protocol_native_endpoint_stream_method_demarshal[PW_ENDPOINT_STREAM_PROXY_METHOD_NUM] =
4709 + [PW_ENDPOINT_STREAM_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
4710 + [PW_ENDPOINT_STREAM_PROXY_METHOD_SUBSCRIBE_PARAMS] = { endpoint_stream_demarshal_subscribe_params, 0 },
4711 + [PW_ENDPOINT_STREAM_PROXY_METHOD_ENUM_PARAMS] = { endpoint_stream_demarshal_enum_params, 0 },
4712 + [PW_ENDPOINT_STREAM_PROXY_METHOD_SET_PARAM] = { endpoint_stream_demarshal_set_param, PW_PERM_W },
4715 +static const struct pw_protocol_marshal pw_protocol_native_endpoint_stream_marshal = {
4716 + PW_TYPE_INTERFACE_EndpointStream,
4717 + PW_VERSION_ENDPOINT_STREAM_PROXY,
4718 + PW_ENDPOINT_STREAM_PROXY_METHOD_NUM,
4719 + PW_ENDPOINT_STREAM_PROXY_EVENT_NUM,
4720 + &pw_protocol_native_endpoint_stream_method_marshal,
4721 + &pw_protocol_native_endpoint_stream_method_demarshal,
4722 + &pw_protocol_native_endpoint_stream_event_marshal,
4723 + &pw_protocol_native_endpoint_stream_event_demarshal,
4726 +/***********************************************
4728 + ***********************************************/
4730 +static void endpoint_marshal_info (void *object,
4731 + const struct pw_endpoint_info *info)
4733 + struct pw_resource *resource = object;
4734 + struct spa_pod_builder *b;
4736 + b = pw_protocol_native_begin_resource(resource,
4737 + PW_ENDPOINT_PROXY_EVENT_INFO, NULL);
4739 + marshal_pw_endpoint_info(b, info);
4741 + pw_protocol_native_end_resource(resource, b);
4744 +static void endpoint_marshal_param (void *object, int seq, uint32_t id,
4745 + uint32_t index, uint32_t next,
4746 + const struct spa_pod *param)
4748 + struct pw_resource *resource = object;
4749 + struct spa_pod_builder *b;
4751 + b = pw_protocol_native_begin_resource(resource,
4752 + PW_ENDPOINT_PROXY_EVENT_PARAM, NULL);
4754 + spa_pod_builder_add_struct(b,
4757 + SPA_POD_Int(index),
4758 + SPA_POD_Int(next),
4759 + SPA_POD_Pod(param));
4761 + pw_protocol_native_end_resource(resource, b);
4764 +static int endpoint_marshal_add_listener(void *object,
4765 + struct spa_hook *listener,
4766 + const struct pw_endpoint_proxy_events *events,
4769 + struct pw_proxy *proxy = object;
4770 + pw_proxy_add_object_listener(proxy, listener, events, data);
4774 +static int endpoint_marshal_subscribe_params(void *object,
4775 + uint32_t *ids, uint32_t n_ids)
4777 + struct pw_proxy *proxy = object;
4778 + struct spa_pod_builder *b;
4780 + b = pw_protocol_native_begin_proxy(proxy,
4781 + PW_ENDPOINT_PROXY_METHOD_SUBSCRIBE_PARAMS, NULL);
4783 + spa_pod_builder_add_struct(b,
4784 + SPA_POD_Array(sizeof(uint32_t), SPA_TYPE_Id, n_ids, ids));
4786 + return pw_protocol_native_end_proxy(proxy, b);
4789 +static int endpoint_marshal_enum_params(void *object,
4790 + int seq, uint32_t id,
4791 + uint32_t start, uint32_t num,
4792 + const struct spa_pod *filter)
4794 + struct pw_protocol_native_message *msg;
4795 + struct pw_proxy *proxy = object;
4796 + struct spa_pod_builder *b;
4798 + b = pw_protocol_native_begin_proxy(proxy,
4799 + PW_ENDPOINT_PROXY_METHOD_ENUM_PARAMS, &msg);
4801 + spa_pod_builder_add_struct(b,
4802 + SPA_POD_Int(SPA_RESULT_RETURN_ASYNC(msg->seq)),
4804 + SPA_POD_Int(index),
4806 + SPA_POD_Pod(filter));
4808 + return pw_protocol_native_end_proxy(proxy, b);
4811 +static int endpoint_marshal_set_param(void *object,
4812 + uint32_t id, uint32_t flags,
4813 + const struct spa_pod *param)
4815 + struct pw_proxy *proxy = object;
4816 + struct spa_pod_builder *b;
4818 + b = pw_protocol_native_begin_proxy(proxy,
4819 + PW_ENDPOINT_PROXY_METHOD_SET_PARAM, NULL);
4821 + spa_pod_builder_add_struct(b,
4823 + SPA_POD_Int(flags),
4824 + SPA_POD_Pod(param));
4826 + return pw_protocol_native_end_proxy(proxy, b);
4829 +static int endpoint_demarshal_info(void *object,
4830 + const struct pw_protocol_native_message *msg)
4832 + struct pw_proxy *proxy = object;
4833 + struct spa_pod_parser prs;
4834 + struct spa_pod_frame f;
4835 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
4836 + struct pw_endpoint_info info = { .props = &props };
4838 + spa_pod_parser_init(&prs, msg->data, msg->size);
4840 + demarshal_pw_endpoint_info(&prs, &f, &info);
4842 + return pw_proxy_notify(proxy, struct pw_endpoint_proxy_events,
4846 +static int endpoint_demarshal_param(void *object,
4847 + const struct pw_protocol_native_message *msg)
4849 + struct pw_proxy *proxy = object;
4850 + struct spa_pod_parser prs;
4851 + uint32_t id, index, next;
4853 + struct spa_pod *param;
4855 + spa_pod_parser_init(&prs, msg->data, msg->size);
4856 + if (spa_pod_parser_get_struct(&prs,
4857 + SPA_POD_Int(&seq),
4859 + SPA_POD_Int(&index),
4860 + SPA_POD_Int(&next),
4861 + SPA_POD_Pod(¶m)) < 0)
4864 + return pw_proxy_notify(proxy, struct pw_endpoint_proxy_events,
4865 + param, 0, seq, id, index, next, param);
4868 +static int endpoint_demarshal_subscribe_params(void *object,
4869 + const struct pw_protocol_native_message *msg)
4871 + struct pw_resource *resource = object;
4872 + struct spa_pod_parser prs;
4873 + uint32_t csize, ctype, n_ids;
4876 + spa_pod_parser_init(&prs, msg->data, msg->size);
4877 + if (spa_pod_parser_get_struct(&prs,
4878 + SPA_POD_Array(&csize, &ctype, &n_ids, &ids)) < 0)
4881 + if (ctype != SPA_TYPE_Id)
4884 + return pw_resource_notify(resource, struct pw_endpoint_proxy_methods,
4885 + subscribe_params, 0, ids, n_ids);
4888 +static int endpoint_demarshal_enum_params(void *object,
4889 + const struct pw_protocol_native_message *msg)
4891 + struct pw_resource *resource = object;
4892 + struct spa_pod_parser prs;
4893 + uint32_t id, index, num;
4895 + struct spa_pod *filter;
4897 + spa_pod_parser_init(&prs, msg->data, msg->size);
4898 + if (spa_pod_parser_get_struct(&prs,
4899 + SPA_POD_Int(&seq),
4901 + SPA_POD_Int(&index),
4902 + SPA_POD_Int(&num),
4903 + SPA_POD_Pod(&filter)) < 0)
4906 + return pw_resource_notify(resource, struct pw_endpoint_proxy_methods,
4907 + enum_params, 0, seq, id, index, num, filter);
4910 +static int endpoint_demarshal_set_param(void *object,
4911 + const struct pw_protocol_native_message *msg)
4913 + struct pw_resource *resource = object;
4914 + struct spa_pod_parser prs;
4915 + uint32_t id, flags;
4916 + struct spa_pod *param;
4918 + spa_pod_parser_init(&prs, msg->data, msg->size);
4919 + if (spa_pod_parser_get_struct(&prs,
4921 + SPA_POD_Int(&flags),
4922 + SPA_POD_Pod(¶m)) < 0)
4925 + return pw_resource_notify(resource, struct pw_endpoint_proxy_methods,
4926 + set_param, 0, id, flags, param);
4929 +static const struct pw_endpoint_proxy_events pw_protocol_native_endpoint_event_marshal = {
4930 + PW_VERSION_ENDPOINT_PROXY_EVENTS,
4931 + .info = endpoint_marshal_info,
4932 + .param = endpoint_marshal_param,
4935 +static const struct pw_protocol_native_demarshal
4936 +pw_protocol_native_endpoint_event_demarshal[PW_ENDPOINT_PROXY_EVENT_NUM] =
4938 + [PW_ENDPOINT_PROXY_EVENT_INFO] = { endpoint_demarshal_info, 0 },
4939 + [PW_ENDPOINT_PROXY_EVENT_PARAM] = { endpoint_demarshal_param, 0 },
4942 +static const struct pw_endpoint_proxy_methods pw_protocol_native_endpoint_method_marshal = {
4943 + PW_VERSION_ENDPOINT_PROXY_METHODS,
4944 + .add_listener = endpoint_marshal_add_listener,
4945 + .subscribe_params = endpoint_marshal_subscribe_params,
4946 + .enum_params = endpoint_marshal_enum_params,
4947 + .set_param = endpoint_marshal_set_param,
4950 +static const struct pw_protocol_native_demarshal
4951 +pw_protocol_native_endpoint_method_demarshal[PW_ENDPOINT_PROXY_METHOD_NUM] =
4953 + [PW_ENDPOINT_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
4954 + [PW_ENDPOINT_PROXY_METHOD_SUBSCRIBE_PARAMS] = { endpoint_demarshal_subscribe_params, 0 },
4955 + [PW_ENDPOINT_PROXY_METHOD_ENUM_PARAMS] = { endpoint_demarshal_enum_params, 0 },
4956 + [PW_ENDPOINT_PROXY_METHOD_SET_PARAM] = { endpoint_demarshal_set_param, PW_PERM_W },
4959 +static const struct pw_protocol_marshal pw_protocol_native_endpoint_marshal = {
4960 + PW_TYPE_INTERFACE_Endpoint,
4961 + PW_VERSION_ENDPOINT_PROXY,
4962 + PW_ENDPOINT_PROXY_METHOD_NUM,
4963 + PW_ENDPOINT_PROXY_EVENT_NUM,
4964 + &pw_protocol_native_endpoint_method_marshal,
4965 + &pw_protocol_native_endpoint_method_demarshal,
4966 + &pw_protocol_native_endpoint_event_marshal,
4967 + &pw_protocol_native_endpoint_event_demarshal,
4970 +/***********************************************
4972 + ***********************************************/
4974 +static void session_marshal_info (void *object,
4975 + const struct pw_session_info *info)
4977 + struct pw_resource *resource = object;
4978 + struct spa_pod_builder *b;
4980 + b = pw_protocol_native_begin_resource(resource,
4981 + PW_SESSION_PROXY_EVENT_INFO, NULL);
4983 + marshal_pw_session_info(b, info);
4985 + pw_protocol_native_end_resource(resource, b);
4988 +static void session_marshal_param (void *object, int seq, uint32_t id,
4989 + uint32_t index, uint32_t next,
4990 + const struct spa_pod *param)
4992 + struct pw_resource *resource = object;
4993 + struct spa_pod_builder *b;
4995 + b = pw_protocol_native_begin_resource(resource,
4996 + PW_SESSION_PROXY_EVENT_PARAM, NULL);
4998 + spa_pod_builder_add_struct(b,
5001 + SPA_POD_Int(index),
5002 + SPA_POD_Int(next),
5003 + SPA_POD_Pod(param));
5005 + pw_protocol_native_end_resource(resource, b);
5008 +static int session_marshal_add_listener(void *object,
5009 + struct spa_hook *listener,
5010 + const struct pw_session_proxy_events *events,
5013 + struct pw_proxy *proxy = object;
5014 + pw_proxy_add_object_listener(proxy, listener, events, data);
5018 +static int session_marshal_subscribe_params(void *object,
5019 + uint32_t *ids, uint32_t n_ids)
5021 + struct pw_proxy *proxy = object;
5022 + struct spa_pod_builder *b;
5024 + b = pw_protocol_native_begin_proxy(proxy,
5025 + PW_SESSION_PROXY_METHOD_SUBSCRIBE_PARAMS, NULL);
5027 + spa_pod_builder_add_struct(b,
5028 + SPA_POD_Array(sizeof(uint32_t), SPA_TYPE_Id, n_ids, ids));
5030 + return pw_protocol_native_end_proxy(proxy, b);
5033 +static int session_marshal_enum_params(void *object,
5034 + int seq, uint32_t id,
5035 + uint32_t start, uint32_t num,
5036 + const struct spa_pod *filter)
5038 + struct pw_protocol_native_message *msg;
5039 + struct pw_proxy *proxy = object;
5040 + struct spa_pod_builder *b;
5042 + b = pw_protocol_native_begin_proxy(proxy,
5043 + PW_SESSION_PROXY_METHOD_ENUM_PARAMS, &msg);
5045 + spa_pod_builder_add_struct(b,
5046 + SPA_POD_Int(SPA_RESULT_RETURN_ASYNC(msg->seq)),
5048 + SPA_POD_Int(index),
5050 + SPA_POD_Pod(filter));
5052 + return pw_protocol_native_end_proxy(proxy, b);
5055 +static int session_marshal_set_param(void *object,
5056 + uint32_t id, uint32_t flags,
5057 + const struct spa_pod *param)
5059 + struct pw_proxy *proxy = object;
5060 + struct spa_pod_builder *b;
5062 + b = pw_protocol_native_begin_proxy(proxy,
5063 + PW_SESSION_PROXY_METHOD_SET_PARAM, NULL);
5065 + spa_pod_builder_add_struct(b,
5067 + SPA_POD_Int(flags),
5068 + SPA_POD_Pod(param));
5070 + return pw_protocol_native_end_proxy(proxy, b);
5073 +static int session_marshal_create_link(void *object,
5074 + const struct spa_dict *props)
5076 + struct pw_proxy *proxy = object;
5077 + struct spa_pod_builder *b;
5079 + b = pw_protocol_native_begin_proxy(proxy,
5080 + PW_SESSION_PROXY_METHOD_CREATE_LINK, NULL);
5082 + push_dict(b, props);
5084 + return pw_protocol_native_end_proxy(proxy, b);
5087 +static int session_demarshal_info(void *object,
5088 + const struct pw_protocol_native_message *msg)
5090 + struct pw_proxy *proxy = object;
5091 + struct spa_pod_parser prs;
5092 + struct spa_pod_frame f;
5093 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
5094 + struct pw_session_info info = { .props = &props };
5096 + spa_pod_parser_init(&prs, msg->data, msg->size);
5098 + demarshal_pw_session_info(&prs, &f, &info);
5100 + return pw_proxy_notify(proxy, struct pw_session_proxy_events,
5104 +static int session_demarshal_param(void *object,
5105 + const struct pw_protocol_native_message *msg)
5107 + struct pw_proxy *proxy = object;
5108 + struct spa_pod_parser prs;
5109 + uint32_t id, index, next;
5111 + struct spa_pod *param;
5113 + spa_pod_parser_init(&prs, msg->data, msg->size);
5114 + if (spa_pod_parser_get_struct(&prs,
5115 + SPA_POD_Int(&seq),
5117 + SPA_POD_Int(&index),
5118 + SPA_POD_Int(&next),
5119 + SPA_POD_Pod(¶m)) < 0)
5122 + return pw_proxy_notify(proxy, struct pw_session_proxy_events,
5123 + param, 0, seq, id, index, next, param);
5126 +static int session_demarshal_subscribe_params(void *object,
5127 + const struct pw_protocol_native_message *msg)
5129 + struct pw_resource *resource = object;
5130 + struct spa_pod_parser prs;
5131 + uint32_t csize, ctype, n_ids;
5134 + spa_pod_parser_init(&prs, msg->data, msg->size);
5135 + if (spa_pod_parser_get_struct(&prs,
5136 + SPA_POD_Array(&csize, &ctype, &n_ids, &ids)) < 0)
5139 + if (ctype != SPA_TYPE_Id)
5142 + return pw_resource_notify(resource, struct pw_session_proxy_methods,
5143 + subscribe_params, 0, ids, n_ids);
5146 +static int session_demarshal_enum_params(void *object,
5147 + const struct pw_protocol_native_message *msg)
5149 + struct pw_resource *resource = object;
5150 + struct spa_pod_parser prs;
5151 + uint32_t id, index, num;
5153 + struct spa_pod *filter;
5155 + spa_pod_parser_init(&prs, msg->data, msg->size);
5156 + if (spa_pod_parser_get_struct(&prs,
5157 + SPA_POD_Int(&seq),
5159 + SPA_POD_Int(&index),
5160 + SPA_POD_Int(&num),
5161 + SPA_POD_Pod(&filter)) < 0)
5164 + return pw_resource_notify(resource, struct pw_session_proxy_methods,
5165 + enum_params, 0, seq, id, index, num, filter);
5168 +static int session_demarshal_set_param(void *object,
5169 + const struct pw_protocol_native_message *msg)
5171 + struct pw_resource *resource = object;
5172 + struct spa_pod_parser prs;
5173 + uint32_t id, flags;
5174 + struct spa_pod *param;
5176 + spa_pod_parser_init(&prs, msg->data, msg->size);
5177 + if (spa_pod_parser_get_struct(&prs,
5179 + SPA_POD_Int(&flags),
5180 + SPA_POD_Pod(¶m)) < 0)
5183 + return pw_resource_notify(resource, struct pw_session_proxy_methods,
5184 + set_param, 0, id, flags, param);
5187 +static int session_demarshal_create_link(void *object,
5188 + const struct pw_protocol_native_message *msg)
5190 + struct pw_resource *resource = object;
5191 + struct spa_pod_parser prs;
5192 + struct spa_pod_frame f;
5193 + struct spa_dict props = SPA_DICT_INIT(NULL, 0);
5195 + spa_pod_parser_init(&prs, msg->data, msg->size);
5197 + parse_dict(&prs, &f, &props);
5199 + return pw_resource_notify(resource, struct pw_session_proxy_methods,
5200 + create_link, 0, &props);
5203 +static const struct pw_session_proxy_events pw_protocol_native_session_event_marshal = {
5204 + PW_VERSION_SESSION_PROXY_EVENTS,
5205 + .info = session_marshal_info,
5206 + .param = session_marshal_param,
5209 +static const struct pw_protocol_native_demarshal
5210 +pw_protocol_native_session_event_demarshal[PW_SESSION_PROXY_EVENT_NUM] =
5212 + [PW_SESSION_PROXY_EVENT_INFO] = { session_demarshal_info, 0 },
5213 + [PW_SESSION_PROXY_EVENT_PARAM] = { session_demarshal_param, 0 },
5216 +static const struct pw_session_proxy_methods pw_protocol_native_session_method_marshal = {
5217 + PW_VERSION_SESSION_PROXY_METHODS,
5218 + .add_listener = session_marshal_add_listener,
5219 + .subscribe_params = session_marshal_subscribe_params,
5220 + .enum_params = session_marshal_enum_params,
5221 + .set_param = session_marshal_set_param,
5222 + .create_link = session_marshal_create_link,
5225 +static const struct pw_protocol_native_demarshal
5226 +pw_protocol_native_session_method_demarshal[PW_SESSION_PROXY_METHOD_NUM] =
5228 + [PW_SESSION_PROXY_METHOD_ADD_LISTENER] = { NULL, 0 },
5229 + [PW_SESSION_PROXY_METHOD_SUBSCRIBE_PARAMS] = { session_demarshal_subscribe_params, 0 },
5230 + [PW_SESSION_PROXY_METHOD_ENUM_PARAMS] = { session_demarshal_enum_params, 0 },
5231 + [PW_SESSION_PROXY_METHOD_SET_PARAM] = { session_demarshal_set_param, PW_PERM_W },
5232 + [PW_SESSION_PROXY_METHOD_CREATE_LINK] = { session_demarshal_create_link, PW_PERM_W },
5235 +static const struct pw_protocol_marshal pw_protocol_native_session_marshal = {
5236 + PW_TYPE_INTERFACE_Session,
5237 + PW_VERSION_SESSION_PROXY,
5238 + PW_SESSION_PROXY_METHOD_NUM,
5239 + PW_SESSION_PROXY_EVENT_NUM,
5240 + &pw_protocol_native_session_method_marshal,
5241 + &pw_protocol_native_session_method_demarshal,
5242 + &pw_protocol_native_session_event_marshal,
5243 + &pw_protocol_native_session_event_demarshal,
5246 +struct pw_protocol *pw_protocol_native_ext_session_manager_init(struct pw_core *core)
5248 + struct pw_protocol *protocol;
5250 + protocol = pw_core_find_protocol(core, PW_TYPE_INFO_PROTOCOL_Native);
5252 + if (protocol == NULL)
5255 + pw_protocol_add_marshal(protocol, &pw_protocol_native_client_endpoint_marshal);
5256 + pw_protocol_add_marshal(protocol, &pw_protocol_native_client_session_marshal);
5257 + pw_protocol_add_marshal(protocol, &pw_protocol_native_endpoint_link_marshal);
5258 + pw_protocol_add_marshal(protocol, &pw_protocol_native_endpoint_stream_marshal);
5259 + pw_protocol_add_marshal(protocol, &pw_protocol_native_endpoint_marshal);
5260 + pw_protocol_add_marshal(protocol, &pw_protocol_native_session_marshal);
5264 diff --git a/src/modules/module-session-manager/session.c b/src/modules/module-session-manager/session.c
5265 new file mode 100644
5266 index 00000000..226eba4e
5268 +++ b/src/modules/module-session-manager/session.c
5272 + * Copyright © 2019 Collabora Ltd.
5273 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
5275 + * Permission is hereby granted, free of charge, to any person obtaining a
5276 + * copy of this software and associated documentation files (the "Software"),
5277 + * to deal in the Software without restriction, including without limitation
5278 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
5279 + * and/or sell copies of the Software, and to permit persons to whom the
5280 + * Software is furnished to do so, subject to the following conditions:
5282 + * The above copyright notice and this permission notice (including the next
5283 + * paragraph) shall be included in all copies or substantial portions of the
5286 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
5287 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
5288 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
5289 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
5290 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
5291 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
5292 + * DEALINGS IN THE SOFTWARE.
5295 +#include <stdbool.h>
5296 +#include <string.h>
5298 +#include <pipewire/pipewire.h>
5299 +#include <extensions/session-manager.h>
5301 +#include <spa/pod/filter.h>
5303 +#include "session.h"
5304 +#include "client-session.h"
5306 +#include <pipewire/private.h>
5308 +#define NAME "session"
5310 +struct resource_data {
5311 + struct session *session;
5312 + struct spa_hook resource_listener;
5313 + struct spa_hook object_listener;
5314 + uint32_t n_subscribe_ids;
5315 + uint32_t subscribe_ids[32];
5318 +#define pw_session_resource(r,m,v,...) \
5319 + pw_resource_call(r,struct pw_session_proxy_events,m,v,__VA_ARGS__)
5320 +#define pw_session_resource_info(r,...) \
5321 + pw_session_resource(r,info,0,__VA_ARGS__)
5322 +#define pw_session_resource_param(r,...) \
5323 + pw_session_resource(r,param,0,__VA_ARGS__)
5325 +static int session_enum_params (void *object, int seq,
5326 + uint32_t id, uint32_t start, uint32_t num,
5327 + const struct spa_pod *filter)
5329 + struct pw_resource *resource = object;
5330 + struct resource_data *data = pw_resource_get_user_data(resource);
5331 + struct session *this = data->session;
5332 + struct spa_pod *result;
5333 + struct spa_pod *param;
5334 + uint8_t buffer[1024];
5335 + struct spa_pod_builder b = { 0 };
5337 + uint32_t next = start;
5338 + uint32_t count = 0;
5342 + if (index >= this->n_params)
5345 + param = this->params[index];
5347 + if (param == NULL || !spa_pod_is_object_id(param, id))
5350 + spa_pod_builder_init(&b, buffer, sizeof(buffer));
5351 + if (spa_pod_filter(&b, &result, param, filter) != 0)
5354 + pw_log_debug(NAME" %p: %d param %u", this, seq, index);
5356 + pw_session_resource_param(resource, seq, id, index, next, result);
5358 + if (++count == num)
5364 +static int session_subscribe_params (void *object, uint32_t *ids, uint32_t n_ids)
5366 + struct pw_resource *resource = object;
5367 + struct resource_data *data = pw_resource_get_user_data(resource);
5370 + n_ids = SPA_MIN(n_ids, SPA_N_ELEMENTS(data->subscribe_ids));
5371 + data->n_subscribe_ids = n_ids;
5373 + for (i = 0; i < n_ids; i++) {
5374 + data->subscribe_ids[i] = ids[i];
5375 + pw_log_debug(NAME" %p: resource %d subscribe param %u",
5376 + data->session, resource->id, ids[i]);
5377 + session_enum_params(resource, 1, ids[i], 0, UINT32_MAX, NULL);
5382 +static int session_set_param (void *object, uint32_t id, uint32_t flags,
5383 + const struct spa_pod *param)
5385 + struct pw_resource *resource = object;
5386 + struct resource_data *data = pw_resource_get_user_data(resource);
5387 + struct session *this = data->session;
5389 + pw_client_session_resource_set_param(this->client_sess->resource,
5390 + id, flags, param);
5395 +static int session_create_link(void *object, const struct spa_dict *props)
5397 + struct pw_resource *resource = object;
5398 + struct resource_data *data = pw_resource_get_user_data(resource);
5399 + struct session *this = data->session;
5401 + pw_client_session_resource_create_link(this->client_sess->resource,
5407 +static const struct pw_session_proxy_methods methods = {
5408 + PW_VERSION_SESSION_PROXY_METHODS,
5409 + .subscribe_params = session_subscribe_params,
5410 + .enum_params = session_enum_params,
5411 + .set_param = session_set_param,
5412 + .create_link = session_create_link,
5415 +static void session_notify_subscribed(struct session *this,
5416 + uint32_t index, uint32_t next)
5418 + struct pw_global *global = this->global;
5419 + struct pw_resource *resource;
5420 + struct resource_data *data;
5421 + struct spa_pod *param = this->params[index];
5425 + if (!param || !spa_pod_is_object (param))
5428 + id = SPA_POD_OBJECT_ID (param);
5430 + spa_list_for_each(resource, &global->resource_list, link) {
5431 + data = pw_resource_get_user_data(resource);
5432 + for (i = 0; i < data->n_subscribe_ids; i++) {
5433 + if (data->subscribe_ids[i] == id) {
5434 + pw_session_resource_param(resource, 1, id,
5435 + index, next, param);
5441 +int session_update(struct session *this,
5442 + uint32_t change_mask,
5443 + uint32_t n_params,
5444 + const struct spa_pod **params,
5445 + const struct pw_session_info *info)
5447 + if (change_mask & PW_CLIENT_SESSION_UPDATE_PARAMS) {
5449 + size_t size = n_params * sizeof(struct spa_pod *);
5451 + pw_log_debug(NAME" %p: update %d params", this, n_params);
5453 + for (i = 0; i < this->n_params; i++)
5454 + free(this->params[i]);
5455 + this->params = realloc(this->params, size);
5456 + if (size > 0 && !this->params) {
5457 + this->n_params = 0;
5460 + this->n_params = n_params;
5462 + for (i = 0; i < this->n_params; i++) {
5463 + this->params[i] = params[i] ? spa_pod_copy(params[i]) : NULL;
5464 + session_notify_subscribed(this, i, i+1);
5468 + if (change_mask & PW_CLIENT_SESSION_UPDATE_INFO) {
5469 + struct pw_resource *resource;
5471 + if (info->change_mask & PW_SESSION_CHANGE_MASK_PROPS)
5472 + pw_properties_update(this->props, info->props);
5474 + if (info->change_mask & PW_SESSION_CHANGE_MASK_PARAMS) {
5475 + size_t size = info->n_params * sizeof(struct spa_param_info);
5477 + this->info.params = realloc(this->info.params, size);
5478 + if (size > 0 && !this->info.params) {
5479 + this->info.n_params = 0;
5482 + this->info.n_params = info->n_params;
5484 + memcpy(this->info.params, info->params, size);
5487 + this->info.change_mask = info->change_mask;
5488 + spa_list_for_each(resource, &this->global->resource_list, link) {
5489 + pw_session_resource_info(resource, &this->info);
5491 + this->info.change_mask = 0;
5497 + pw_log_error(NAME" can't update: no memory");
5498 + pw_resource_error(this->client_sess->resource, -ENOMEM,
5499 + NAME" can't update: no memory");
5503 +static void session_unbind(void *data)
5505 + struct pw_resource *resource = data;
5506 + spa_list_remove(&resource->link);
5509 +static const struct pw_resource_events resource_events = {
5510 + PW_VERSION_RESOURCE_EVENTS,
5511 + .destroy = session_unbind,
5514 +static int session_bind(void *_data, struct pw_client *client,
5515 + uint32_t permissions, uint32_t version, uint32_t id)
5517 + struct session *this = _data;
5518 + struct pw_global *global = this->global;
5519 + struct pw_resource *resource;
5520 + struct resource_data *data;
5522 + resource = pw_resource_new(client, id, permissions, global->type, version, sizeof(*data));
5523 + if (resource == NULL)
5526 + data = pw_resource_get_user_data(resource);
5527 + data->session = this;
5528 + pw_resource_add_listener(resource, &data->resource_listener,
5529 + &resource_events, resource);
5530 + pw_resource_add_object_listener(resource, &data->object_listener,
5531 + &methods, resource);
5533 + pw_log_debug(NAME" %p: bound to %d", this, resource->id);
5535 + spa_list_append(&global->resource_list, &resource->link);
5537 + this->info.change_mask = PW_SESSION_CHANGE_MASK_ALL;
5538 + pw_session_resource_info(resource, &this->info);
5539 + this->info.change_mask = 0;
5544 + pw_log_error(NAME" can't create resource: no memory");
5545 + pw_resource_error(this->client_sess->resource, -ENOMEM,
5546 + NAME" can't create resource: no memory");
5550 +int session_init(struct session *this,
5551 + struct client_session *client_sess,
5552 + struct pw_core *core,
5553 + struct pw_properties *properties)
5555 + const char *keys[] = {
5556 + PW_KEY_FACTORY_ID,
5561 + pw_log_debug(NAME" %p: new", this);
5563 + this->client_sess = client_sess;
5564 + this->props = properties;
5566 + properties = pw_properties_new(NULL, NULL);
5570 + pw_properties_copy_keys(this->props, properties, keys);
5572 + this->global = pw_global_new (core,
5573 + PW_TYPE_INTERFACE_Session,
5574 + PW_VERSION_SESSION_PROXY,
5575 + properties, session_bind, this);
5576 + if (!this->global)
5579 + pw_properties_setf(this->props, PW_KEY_SESSION_ID, "%u", this->global->id);
5581 + this->info.version = PW_VERSION_SESSION_INFO;
5582 + this->info.id = this->global->id;
5583 + this->info.props = &this->props->dict;
5585 + pw_client_session_resource_set_id(client_sess->resource, this->global->id);
5587 + return pw_global_register(this->global);
5590 + pw_log_error(NAME" - can't create - out of memory");
5594 +void session_clear(struct session *this)
5598 + pw_log_debug(NAME" %p: destroy", this);
5600 + pw_global_destroy(this->global);
5602 + for (i = 0; i < this->n_params; i++)
5603 + free(this->params[i]);
5604 + free(this->params);
5606 + free(this->info.params);
5609 + pw_properties_free(this->props);
5611 diff --git a/src/modules/module-session-manager/session.h b/src/modules/module-session-manager/session.h
5612 new file mode 100644
5613 index 00000000..ad0b9b1b
5615 +++ b/src/modules/module-session-manager/session.h
5619 + * Copyright © 2019 Collabora Ltd.
5620 + * @author George Kiagiadakis <george.kiagiadakis@collabora.com>
5622 + * Permission is hereby granted, free of charge, to any person obtaining a
5623 + * copy of this software and associated documentation files (the "Software"),
5624 + * to deal in the Software without restriction, including without limitation
5625 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
5626 + * and/or sell copies of the Software, and to permit persons to whom the
5627 + * Software is furnished to do so, subject to the following conditions:
5629 + * The above copyright notice and this permission notice (including the next
5630 + * paragraph) shall be included in all copies or substantial portions of the
5633 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
5634 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
5635 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
5636 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
5637 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
5638 + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
5639 + * DEALINGS IN THE SOFTWARE.
5642 +#ifndef MODULE_SESSION_MANAGER_SESSION_H
5643 +#define MODULE_SESSION_MANAGER_SESSION_H
5649 +struct client_session;
5652 + struct client_session *client_sess;
5653 + struct pw_global *global;
5654 + uint32_t n_params;
5655 + struct spa_pod **params;
5656 + struct pw_session_info info;
5657 + struct pw_properties *props; /* wrapper of info.props */
5660 +int session_init(struct session *this,
5661 + struct client_session *client_sess,
5662 + struct pw_core *core,
5663 + struct pw_properties *properties);
5665 +void session_clear(struct session *this);
5667 +int session_update(struct session *this,
5668 + uint32_t change_mask,
5669 + uint32_t n_params,
5670 + const struct spa_pod **params,
5671 + const struct pw_session_info *info);
5677 +#endif /* MODULE_SESSION_MANAGER_SESSION_H */
5678 diff --git a/src/pipewire/pipewire.c b/src/pipewire/pipewire.c
5679 index a7012ab2..ec5f1f85 100644
5680 --- a/src/pipewire/pipewire.c
5681 +++ b/src/pipewire/pipewire.c
5682 @@ -575,6 +575,12 @@ static const struct spa_type_info type_info[] = {
5683 { PW_TYPE_INTERFACE_Module, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "Module", NULL },
5684 { PW_TYPE_INTERFACE_ClientNode, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "ClientNode", NULL },
5685 { PW_TYPE_INTERFACE_Device, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "Device", NULL },
5686 + { PW_TYPE_INTERFACE_ClientEndpoint, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "ClientEndpoint", NULL},
5687 + { PW_TYPE_INTERFACE_Endpoint, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "Endpoint", NULL},
5688 + { PW_TYPE_INTERFACE_EndpointStream, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "EndpointStream", NULL},
5689 + { PW_TYPE_INTERFACE_ClientSession, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "ClientSession", NULL},
5690 + { PW_TYPE_INTERFACE_Session, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "Session", NULL},
5691 + { PW_TYPE_INTERFACE_EndpointLink, SPA_TYPE_Pointer, PW_TYPE_INFO_INTERFACE_BASE "EndpointLink", NULL},
5692 { SPA_ID_INVALID, SPA_ID_INVALID, "spa_types", spa_types },
5693 { 0, 0, NULL, NULL },
5695 diff --git a/src/pipewire/type.h b/src/pipewire/type.h
5696 index a1b205f7..6b1b8b50 100644
5697 --- a/src/pipewire/type.h
5698 +++ b/src/pipewire/type.h
5699 @@ -48,7 +48,12 @@ enum {
5701 PW_TYPE_INTERFACE_EXTENSIONS = PW_TYPE_INTERFACE_START + 0x1000,
5702 PW_TYPE_INTERFACE_ClientNode,
5704 + PW_TYPE_INTERFACE_ClientEndpoint,
5705 + PW_TYPE_INTERFACE_Endpoint,
5706 + PW_TYPE_INTERFACE_EndpointStream,
5707 + PW_TYPE_INTERFACE_ClientSession,
5708 + PW_TYPE_INTERFACE_Session,
5709 + PW_TYPE_INTERFACE_EndpointLink,
5712 #define PW_TYPE_INFO_BASE "PipeWire:"