2022-03-22 19:40:23 +01:00
|
|
|
/* AVB support
|
|
|
|
|
*
|
|
|
|
|
* Copyright © 2022 Wim Taymans
|
|
|
|
|
*
|
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
|
|
|
*
|
|
|
|
|
* The above copyright notice and this permission notice (including the next
|
|
|
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
|
|
|
* Software.
|
|
|
|
|
*
|
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
|
* DEALINGS IN THE SOFTWARE.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <pipewire/pipewire.h>
|
|
|
|
|
|
|
|
|
|
#include "mrp.h"
|
|
|
|
|
|
2022-03-23 09:20:38 +01:00
|
|
|
struct attribute {
|
|
|
|
|
struct avbtp_mrp_attribute attr;
|
|
|
|
|
struct spa_list link;
|
|
|
|
|
uint8_t applicant_state;
|
|
|
|
|
uint8_t registrar_state;
|
|
|
|
|
uint16_t pending_indications;
|
|
|
|
|
struct spa_callbacks cb;
|
|
|
|
|
};
|
|
|
|
|
|
2022-03-22 19:40:23 +01:00
|
|
|
struct mrp {
|
|
|
|
|
struct server *server;
|
|
|
|
|
struct spa_hook server_listener;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void mrp_destroy(void *data)
|
|
|
|
|
{
|
|
|
|
|
struct mrp *mrp = data;
|
|
|
|
|
spa_hook_remove(&mrp->server_listener);
|
|
|
|
|
free(mrp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const struct server_events server_events = {
|
|
|
|
|
AVBTP_VERSION_SERVER_EVENTS,
|
|
|
|
|
.destroy = mrp_destroy,
|
|
|
|
|
};
|
|
|
|
|
|
2022-03-23 09:20:38 +01:00
|
|
|
struct avbtp_mrp_attribute *avbtp_mrp_attribute_new(struct avbtp_mrp *mrp,
|
|
|
|
|
uint8_t type, uint8_t param,
|
|
|
|
|
struct avbtp_mrp_attribute_methods *methods, void *data,
|
|
|
|
|
size_t user_size)
|
2022-03-22 19:40:23 +01:00
|
|
|
{
|
2022-03-23 09:20:38 +01:00
|
|
|
struct attribute *a;
|
|
|
|
|
|
|
|
|
|
a = calloc(1, sizeof(*a) + user_size);
|
|
|
|
|
if (a == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
a->attr.type = type;
|
|
|
|
|
a->attr.param = param;
|
|
|
|
|
a->attr.user_data = SPA_PTROFF(a, sizeof(*a), void);
|
|
|
|
|
a->cb = SPA_CALLBACKS_INIT(methods, data);
|
|
|
|
|
|
|
|
|
|
return &a->attr;
|
2022-03-22 19:40:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void stop_avb_timer(void)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
static void start_avb_timer(void)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_update_state(struct avbtp_mrp *mrp,
|
|
|
|
|
struct avbtp_mrp_attribute *attr, int event, uint8_t param)
|
|
|
|
|
{
|
2022-03-23 09:20:38 +01:00
|
|
|
struct attribute *a = SPA_CONTAINER_OF(attr, struct attribute, attr);
|
2022-03-22 19:40:23 +01:00
|
|
|
switch (event) {
|
|
|
|
|
case AVBTP_MRP_EVENT_BEGIN:
|
2022-03-23 09:20:38 +01:00
|
|
|
a->registrar_state = AVBTP_MRP_MT;
|
2022-03-22 19:40:23 +01:00
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_NEW:
|
2022-03-23 09:20:38 +01:00
|
|
|
if (a->registrar_state == AVBTP_MRP_LV)
|
2022-03-22 19:40:23 +01:00
|
|
|
stop_avb_timer();
|
2022-03-23 09:20:38 +01:00
|
|
|
a->registrar_state = AVBTP_MRP_IN;
|
|
|
|
|
a->pending_indications |= AVBTP_PENDING_JOIN_NEW;
|
|
|
|
|
a->attr.param = param;
|
2022-03-22 19:40:23 +01:00
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_JOININ:
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_JOINMT:
|
2022-03-23 09:20:38 +01:00
|
|
|
if (a->registrar_state == AVBTP_MRP_LV)
|
2022-03-22 19:40:23 +01:00
|
|
|
stop_avb_timer();
|
2022-03-23 09:20:38 +01:00
|
|
|
if (a->registrar_state == AVBTP_MRP_MT) {
|
|
|
|
|
a->pending_indications |= AVBTP_PENDING_JOIN;
|
|
|
|
|
a->attr.param = param;
|
2022-03-22 19:40:23 +01:00
|
|
|
}
|
2022-03-23 09:20:38 +01:00
|
|
|
a->registrar_state = AVBTP_MRP_IN;
|
2022-03-22 19:40:23 +01:00
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_LV:
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_LVA:
|
|
|
|
|
case AVBTP_MRP_EVENT_TX_LVA:
|
|
|
|
|
case AVBTP_MRP_EVENT_REDECLARE:
|
2022-03-23 09:20:38 +01:00
|
|
|
if (a->registrar_state == AVBTP_MRP_IN) {
|
2022-03-22 19:40:23 +01:00
|
|
|
start_avb_timer();
|
2022-03-23 09:20:38 +01:00
|
|
|
a->registrar_state = AVBTP_MRP_LV;
|
2022-03-22 19:40:23 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_LV_TIMER:
|
|
|
|
|
case AVBTP_MRP_EVENT_FLUSH:
|
2022-03-23 09:20:38 +01:00
|
|
|
if (a->registrar_state == AVBTP_MRP_LV) {
|
|
|
|
|
a->pending_indications |= AVBTP_PENDING_LEAVE;
|
|
|
|
|
a->attr.param = param;
|
2022-03-22 19:40:23 +01:00
|
|
|
}
|
2022-03-23 09:20:38 +01:00
|
|
|
a->registrar_state = AVBTP_MRP_MT;
|
2022-03-22 19:40:23 +01:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-23 09:20:38 +01:00
|
|
|
switch (event) {
|
|
|
|
|
case AVBTP_MRP_EVENT_BEGIN:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_NEW:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VN;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_JOIN:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VO:
|
|
|
|
|
case AVBTP_MRP_LO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VP;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_LA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AP;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_QO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QP;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_LV:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_VP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_VN:
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_QA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_LA;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_JOININ:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_VP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AP;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QP;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
SPA_FALLTHROUGH;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_IN:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QA;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
SPA_FALLTHROUGH;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_JOINMT:
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_MT:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_QA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_QO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AP;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_LO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VO;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_LV:
|
|
|
|
|
case AVBTP_MRP_EVENT_RX_LVA:
|
|
|
|
|
case AVBTP_MRP_EVENT_REDECLARE:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VO:
|
|
|
|
|
case AVBTP_MRP_AO:
|
|
|
|
|
case AVBTP_MRP_QO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_LO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VN;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_QA:
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VP;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_PERIODIC:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_QA:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AP;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_TX:
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VP:
|
|
|
|
|
case AVBTP_MRP_VN:
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_LA:
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
case AVBTP_MRP_LO:
|
|
|
|
|
// int vector = makeTxEvent(e, st, 0);
|
|
|
|
|
// doTx(st, vector);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_VN:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AN;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QA;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_LA:
|
|
|
|
|
case AVBTP_MRP_LO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_VO;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_EVENT_TX_LVA:
|
|
|
|
|
{
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VP:
|
|
|
|
|
case AVBTP_MRP_VN:
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_LA:
|
|
|
|
|
case AVBTP_MRP_QA:
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
// int vector = makeTxEvent(e, st, 1);
|
|
|
|
|
// doTx(st, vector);
|
|
|
|
|
}
|
|
|
|
|
switch (a->applicant_state) {
|
|
|
|
|
case AVBTP_MRP_VO:
|
|
|
|
|
case AVBTP_MRP_LA:
|
|
|
|
|
case AVBTP_MRP_AO:
|
|
|
|
|
case AVBTP_MRP_QO:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_LO;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_VN:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_AN;
|
|
|
|
|
break;
|
|
|
|
|
case AVBTP_MRP_AN:
|
|
|
|
|
case AVBTP_MRP_AA:
|
|
|
|
|
case AVBTP_MRP_AP:
|
|
|
|
|
case AVBTP_MRP_QP:
|
|
|
|
|
a->applicant_state = AVBTP_MRP_QA;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
2022-03-22 19:40:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_event(struct avbtp_mrp *mrp, struct avbtp_mrp_attribute *attr,
|
|
|
|
|
uint8_t event, uint8_t param)
|
|
|
|
|
{
|
|
|
|
|
static const int map[] = {
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_NEW] = AVBTP_MRP_EVENT_RX_NEW,
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_JOININ] = AVBTP_MRP_EVENT_RX_JOININ,
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_IN] = AVBTP_MRP_EVENT_RX_IN,
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_JOINMT] = AVBTP_MRP_EVENT_RX_JOINMT,
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_MT] = AVBTP_MRP_EVENT_RX_MT,
|
|
|
|
|
[AVBTP_MRP_ATTRIBUTE_EVENT_LV] = AVBTP_MRP_EVENT_RX_LV,
|
|
|
|
|
};
|
|
|
|
|
avbtp_mrp_update_state(mrp, attr, map[event], param);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_mad_begin(struct avbtp_mrp *mrp, struct avbtp_mrp_attribute *attr)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_mad_join(struct avbtp_mrp *mrp, struct avbtp_mrp_attribute *attr, bool is_new)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_mad_leave(struct avbtp_mrp *mrp, struct avbtp_mrp_attribute *attr)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void avbtp_mrp_destroy(struct avbtp_mrp *mrp)
|
|
|
|
|
{
|
|
|
|
|
mrp_destroy(mrp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct avbtp_mrp *avbtp_mrp_new(struct server *server)
|
|
|
|
|
{
|
|
|
|
|
struct mrp *mrp;
|
|
|
|
|
|
|
|
|
|
mrp = calloc(1, sizeof(*mrp));
|
|
|
|
|
if (mrp == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
mrp->server = server;
|
|
|
|
|
|
|
|
|
|
avdecc_server_add_listener(server, &mrp->server_listener, &server_events, mrp);
|
|
|
|
|
|
|
|
|
|
return (struct avbtp_mrp*)mrp;
|
|
|
|
|
}
|