From 6e936f9750448c93ee9e3c8cbf228a9f57c3e9a3 Mon Sep 17 00:00:00 2001 From: NiklasWan Date: Thu, 20 Aug 2020 00:31:49 +0200 Subject: [PATCH] Added virtual ALSA AVB driver --- sound/drivers/Kconfig | 6 + sound/drivers/Makefile | 2 +- sound/drivers/avb/Makefile | 9 + sound/drivers/avb/avb.c | 1270 +++++++++++++++++++++++++++ sound/drivers/avb/avb.h | 39 + sound/drivers/avb/avb_util.c | 135 +++ sound/drivers/avb/avb_util.h | 307 +++++++ sound/drivers/avb/avdecc.c | 1602 ++++++++++++++++++++++++++++++++++ sound/drivers/avb/avdecc.h | 454 ++++++++++ sound/drivers/avb/avtp.c | 105 +++ sound/drivers/avb/avtp.h | 51 ++ sound/drivers/avb/msrp.c | 391 +++++++++ sound/drivers/avb/msrp.h | 119 +++ 13 files changed, 4489 insertions(+), 1 deletion(-) create mode 100644 sound/drivers/avb/Makefile create mode 100644 sound/drivers/avb/avb.c create mode 100644 sound/drivers/avb/avb.h create mode 100644 sound/drivers/avb/avb_util.c create mode 100644 sound/drivers/avb/avb_util.h create mode 100644 sound/drivers/avb/avdecc.c create mode 100644 sound/drivers/avb/avdecc.h create mode 100644 sound/drivers/avb/avtp.c create mode 100644 sound/drivers/avb/avtp.h create mode 100644 sound/drivers/avb/msrp.c create mode 100644 sound/drivers/avb/msrp.h diff --git a/sound/drivers/Kconfig b/sound/drivers/Kconfig index 09932cc98e9d8..2c31a1b804cac 100644 --- a/sound/drivers/Kconfig +++ b/sound/drivers/Kconfig @@ -1,4 +1,10 @@ # SPDX-License-Identifier: GPL-2.0-only +config SND_AVB + tristate "Generic AVB driver" + select SND_PCM + help + Say 'Y' or 'M' to include support for the PCM AVB device. + config SND_MPU401_UART tristate select SND_RAWMIDI diff --git a/sound/drivers/Makefile b/sound/drivers/Makefile index 615558a281c8f..ed8659dba1191 100644 --- a/sound/drivers/Makefile +++ b/sound/drivers/Makefile @@ -23,4 +23,4 @@ obj-$(CONFIG_SND_MTS64) += snd-mts64.o obj-$(CONFIG_SND_PORTMAN2X4) += snd-portman2x4.o obj-$(CONFIG_SND_ML403_AC97CR) += snd-ml403-ac97cr.o -obj-$(CONFIG_SND) += opl3/ opl4/ mpu401/ vx/ pcsp/ +obj-$(CONFIG_SND) += avb/ opl3/ opl4/ mpu401/ vx/ pcsp/ diff --git a/sound/drivers/avb/Makefile b/sound/drivers/avb/Makefile new file mode 100644 index 0000000000000..5ba4fe2ac8be9 --- /dev/null +++ b/sound/drivers/avb/Makefile @@ -0,0 +1,9 @@ +# SPDX-License-Identifier: GPL-2.0 +# +# Makefile for ALSA +# Copyright (c) 2001 by Jaroslav Kysela +# + +snd-avb-objs := avb_util.o msrp.o avdecc.o avtp.o avb.o + +obj-$(CONFIG_SND_AVB) += snd-avb.o diff --git a/sound/drivers/avb/avb.c b/sound/drivers/avb/avb.c new file mode 100644 index 0000000000000..e2dcd4f87931f --- /dev/null +++ b/sound/drivers/avb/avb.c @@ -0,0 +1,1270 @@ +/* + * Virtual AVB ALSA sound device + * + * Implementation: + * Copyright (c) by Niklas Wantrupp + * + * Based on generic AVB Driver: + * Copyright (c) Indumathi Duraipandian + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include +#include +#include +#include + +#include "avb.h" + +MODULE_AUTHOR("Indumathi Duraipandian "); +MODULE_DESCRIPTION("AVB soundcard"); +MODULE_LICENSE("GPL"); +MODULE_SUPPORTED_DEVICE("{{ALSA,AVB soundcard}}"); + +#define AVB_DEBUG + +static int index[SND_AVB_NUM_CARDS] = SNDRV_DEFAULT_IDX; +static char *id[SND_AVB_NUM_CARDS] = SNDRV_DEFAULT_STR; +static bool enable[SND_AVB_NUM_CARDS] = { 1, + [1 ...(SND_AVB_NUM_CARDS - 1)] = 0 }; +static int pcm_substreams[SND_AVB_NUM_CARDS] = { 1 }; +static int pcm_notify[SND_AVB_NUM_CARDS]; + +module_param_array(index, int, NULL, 0444); +MODULE_PARM_DESC(index, "Index value for avb soundcard."); +module_param_array(id, charp, NULL, 0444); +MODULE_PARM_DESC(id, "ID string for avb soundcard."); +module_param_array(enable, bool, NULL, 0444); +MODULE_PARM_DESC(enable, "Enable this avb soundcard."); +module_param_array(pcm_substreams, int, NULL, 0444); +MODULE_PARM_DESC(pcm_substreams, "PCM substreams # (1-8) for avb driver."); +module_param_array(pcm_notify, int, NULL, 0444); +MODULE_PARM_DESC(pcm_notify, + "Break capture when PCM format/rate/channels changes."); + +static struct avb_device avb_device; +static int numcards = 0; +static struct platform_device *avbdevices[SND_AVB_NUM_CARDS]; + +static int avb_playback_open(struct snd_pcm_substream *substream); +static int avb_playback_close(struct snd_pcm_substream *substream); +static int avb_playback_hw_params(struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *params); +static int avb_playback_hw_free(struct snd_pcm_substream *substream); +static int avb_playback_prepare(struct snd_pcm_substream *substream); +static int avb_playback_trigger(struct snd_pcm_substream *substream, int cmd); +static snd_pcm_uframes_t +avb_playback_pointer(struct snd_pcm_substream *substream); +static int avb_playback_copy(struct snd_pcm_substream *substream, int channel, + unsigned long pos, void __user *dst, + unsigned long count); +#ifdef AVB_USE_HIGH_RES_TIMER +enum hrtimer_restart avb_avtp_timer(struct hrtimer *t); +#else +static void avb_avtp_timer(unsigned long arg); +#endif + +static int avb_capture_open(struct snd_pcm_substream *substream); +static int avb_capture_close(struct snd_pcm_substream *substream); +static int avb_capture_hw_params(struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *params); +static int avb_capture_hw_free(struct snd_pcm_substream *substream); +static int avb_capture_prepare(struct snd_pcm_substream *substream); +static int avb_capture_trigger(struct snd_pcm_substream *substream, int cmd); +static snd_pcm_uframes_t +avb_capture_pointer(struct snd_pcm_substream *substream); +static int avb_capture_copy(struct snd_pcm_substream *substream, int channel, + unsigned long pos, void __user *dst, + unsigned long count); +static int avb_avtp_listen(struct avb_card *card); + +static void avb_wq_fn(struct work_struct *work); + +static int avb_pcm_new(struct avb_card *avbc, int device, int substreams); + +#ifdef CONFIG_PM_SLEEP + +static int avb_suspend(struct device *pdev); +static int avb_resume(struct device *pdev); + +#endif + +static int avb_probe(struct platform_device *devptr); +static int avb_remove(struct platform_device *devptr); +static void avb_remove_all(void); +static int __init alsa_avb_init(void); +static void __exit alsa_avb_exit(void); + +static SIMPLE_DEV_PM_OPS(avb_pm, avb_suspend, avb_resume); + +static struct platform_driver avb_driver = { + .probe = avb_probe, + .remove = avb_remove, + .driver = { + .name = SND_AVB_DRIVER, + .pm = AVB_PM_OPS, + }, +}; + +static struct snd_pcm_ops avb_playback_ops = { + .open = avb_playback_open, + .close = avb_playback_close, + .ioctl = snd_pcm_lib_ioctl, + .hw_params = avb_playback_hw_params, + .hw_free = avb_playback_hw_free, + .prepare = avb_playback_prepare, + .trigger = avb_playback_trigger, + .pointer = avb_playback_pointer, + .copy_user = avb_playback_copy +}; + +static struct snd_pcm_ops avb_capture_ops = { + .open = avb_capture_open, + .close = avb_capture_close, + .ioctl = snd_pcm_lib_ioctl, + .hw_params = avb_capture_hw_params, + .hw_free = avb_capture_hw_free, + .prepare = avb_capture_prepare, + .trigger = avb_capture_trigger, + .pointer = avb_capture_pointer, + .copy_user = avb_capture_copy +}; + +static struct snd_pcm_hardware avb_playback_hw = { + .info = (SNDRV_PCM_INFO_INTERLEAVED | + SNDRV_PCM_INFO_BLOCK_TRANSFER), + .formats = SNDRV_PCM_FMTBIT_S16_LE, + .rates = SNDRV_PCM_RATE_8000_192000, + .rate_min = 8000, + .rate_max = 192000, + .channels_min = 1, + .channels_max = 8, + .buffer_bytes_max = 131072, + .period_bytes_min = 16384, + .period_bytes_max = 131072, + .periods_min = 2, + .periods_max = 4, +}; + +static struct snd_pcm_hardware avb_capture_hw = { + .info = (SNDRV_PCM_INFO_INTERLEAVED | + SNDRV_PCM_INFO_BLOCK_TRANSFER), + .formats = SNDRV_PCM_FMTBIT_S16_LE, + .rates = SNDRV_PCM_RATE_8000_192000, + .rate_min = 8000, + .rate_max = 192000, + .channels_min = 1, + .channels_max = 8, + .buffer_bytes_max = 131072, + .period_bytes_min = 16384, + .period_bytes_max = 131072, + .periods_min = 2, + .periods_max = 4, +}; + +static struct work_data *avb_init_and_queue_work(int work_id, void *wdata, + int delay) +{ + struct work_data *wd; + + wd = (struct work_data *)kmalloc(sizeof(struct work_data), GFP_KERNEL); + + if (wd == NULL) { + avb_log(AVB_KERN_ERR, + KERN_ERR + "avb_init_and_queue_work work_data allocation failed for work %d", + work_id); + return NULL; + } + + wd->dw.data = wdata; + wd->delayed_work_id = work_id; + INIT_DELAYED_WORK((struct delayed_work *)wd, avb_wq_fn); + + queue_delayed_work(avb_device.wq, (struct delayed_work *)wd, delay); + + return wd; +} + +static int avb_playback_open(struct snd_pcm_substream *substream) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_playback_open"); + + substream->runtime->hw = avb_playback_hw; + + return 0; +} + +static int avb_playback_close(struct snd_pcm_substream *substream) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_playback_close"); + + return 0; +} + +static int avb_playback_hw_params(struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *hw_params) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_log(AVB_KERN_NOT, + KERN_NOTICE "avb_playback_hw_params numbytes:%d sr:%d", + params_buffer_bytes(hw_params), params_rate(hw_params)); + + avb_card->tx.substream = substream; + avb_card->tx.st = 0; + avb_card->tx.sr = params_rate(hw_params); + avb_card->tx.hw_idx = 0; + avb_card->tx.seq_no = 0; + avb_card->tx.hwnw_idx = 0; + avb_card->tx.fill_size = 0; + avb_card->tx.last_timer_ts = jiffies; + avb_card->tx.socket_count = 0; + avb_card->tx.pending_tx_frames = 0; + avb_card->tx.num_bytes_consumed = 0; + avb_card->tx.period_size = params_period_size(hw_params); + avb_card->tx.buffer_size = params_buffer_bytes(hw_params); + avb_card->tx.frame_count = params_buffer_size(hw_params); + avb_card->tx.frame_size = + params_buffer_bytes(hw_params) / params_buffer_size(hw_params); + + avb_avtp_aaf_header_init(&avb_card->sd.tx_buf[0], substream, hw_params); + +#ifdef AVB_USE_HIGH_RES_TIMER + hrtimer_init((struct hrtimer *)&avb_device.txTimer, CLOCK_MONOTONIC, + HRTIMER_MODE_REL); + avb_device.txTimer.timer.function = &avb_avtp_timer; + avb_device.txTimer.card = avb_card; +#else + init_timer(&avb_device.txTimer); + avb_device.txTimer.data = (unsigned long)avb_card; + avb_device.txTimer.function = avb_avtp_timer; + avb_device.txTimer.expires = jiffies + 1; + add_timer(&avb_device.txTimer); +#endif + + memset(&avb_device.tx_ts[0], 0, (sizeof(int) * AVB_MAX_TS_SLOTS)); + avb_device.tx_idx = 0; + + avb_card->tx.tmp_buf = kmalloc(avb_card->tx.buffer_size, GFP_KERNEL); + + return 0; +} + +static int avb_playback_hw_free(struct snd_pcm_substream *substream) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_playback_hw_free"); + +#ifdef AVB_USE_HIGH_RES_TIMER + hrtimer_try_to_cancel((struct hrtimer *)&avb_device.txTimer); +#else + del_timer(&avb_device.txTimer); +#endif + kfree(avb_card->tx.tmp_buf); + + return 0; +} + +static int avb_playback_prepare(struct snd_pcm_substream *substream) +{ + return 0; +} + +static int avb_playback_trigger(struct snd_pcm_substream *substream, int cmd) +{ + ktime_t kt; + int ret = 0; + int avtp_max_frames_per_packet = 0; + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + switch (cmd) { + case SNDRV_PCM_TRIGGER_START: + avb_card->tx.st = 1; + avb_log(AVB_KERN_NOT, + KERN_NOTICE "avb_playback_trigger: Start @ %lu", + jiffies); +#ifdef AVB_USE_HIGH_RES_TIMER + avtp_max_frames_per_packet = + ((ETH_DATA_LEN - sizeof(struct avt_pdu_aaf_pcm_hdr)) / + avb_card->tx.frame_size); + avb_card->tx.timer_val = + ((avtp_max_frames_per_packet * 1000000u) / + (avb_card->tx.sr / 1000)); + kt = ktime_set(0, avb_card->tx.timer_val); + hrtimer_start((struct hrtimer *)&avb_device.txTimer, kt, + HRTIMER_MODE_REL); +#else + avb_card->tx.start_ts = jiffies; + if ((avb_card->tx.pending_tx_frames > 0) && + (!timer_pending(&avb_device.txTimer))) { + avb_device.txTimer.expires = jiffies + 1; + add_timer(&avb_device.txTimer); + } +#endif + break; + case SNDRV_PCM_TRIGGER_STOP: + avb_log(AVB_KERN_NOT, + KERN_NOTICE "avb_playback_trigger: Stop @ %lu", + jiffies); + avb_card->tx.st = 0; +#ifdef AVB_USE_HIGH_RES_TIMER + hrtimer_try_to_cancel((struct hrtimer *)&avb_device.txTimer); +#endif + break; + default: + avb_log(AVB_KERN_WARN, + KERN_WARNING "avb_playback_trigger: Unknown"); + ret = -EINVAL; + } + + return ret; +} + +static snd_pcm_uframes_t +avb_playback_pointer(struct snd_pcm_substream *substream) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_playback_pointer hw_idx:%lu numBytes:%lu, time: %u us", + avb_card->tx.hw_idx, avb_card->tx.num_bytes_consumed, + jiffies_to_usecs(jiffies - avb_card->tx.start_ts)); + + return avb_card->tx.hw_idx; +} + +static int avb_playback_copy(struct snd_pcm_substream *substream, int channel, + unsigned long pos, void __user *dst, + unsigned long count) +{ + int err = 0; + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + count = bytes_to_frames(substream->runtime, count); + pos = bytes_to_frames(substream->runtime, pos); + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_playback_copy: ch:%d, pos: %ld, count: %lu", + channel, pos, count); + + if ((err = copy_from_user( + &avb_card->tx.tmp_buf[(pos * avb_card->tx.frame_size)], + dst, (count * avb_card->tx.frame_size))) != 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_playback_copy copy from user fails: %d \n", + err); + return -1; + } + + avb_card->tx.pending_tx_frames += count; + avb_card->tx.num_bytes_consumed += + (count * (substream->runtime->frame_bits / 8)); + +#ifndef AVB_USE_HIGH_RES_TIMER + if (avb_card->tx.st == 1) { + avb_avtp_timer((unsigned long)avb_card); + } + if ((avb_card->tx.pending_tx_frames > 0) && + (!timer_pending(&avb_device.txTimer) && (avb_card->tx.st == 1))) { + avb_device.txTimer.expires = jiffies + 1; + add_timer(&avb_device.txTimer); + } +#endif + + return 0; +} + +#ifdef AVB_USE_HIGH_RES_TIMER +enum hrtimer_restart avb_avtp_timer(struct hrtimer *t) +#else +static void avb_avtp_timer(unsigned long arg) +#endif +{ + ktime_t kt; + int i = 0; + int err = 0; + int tx_size = 0; + snd_pcm_uframes_t bytes_avai = 0; + snd_pcm_uframes_t bytes_to_copy = 0; + snd_pcm_uframes_t frames_to_copy = 0; + snd_pcm_uframes_t avtp_frames_per_packet = 0; + snd_pcm_uframes_t avtp_max_frames_per_packet = 0; + enum hrtimer_restart hr_res = HRTIMER_NORESTART; +#ifdef AVB_USE_HIGH_RES_TIMER + struct avb_card *avb_card = ((struct avb_hrtimer *)t)->card; +#else + struct avb_card *avb_card = (struct avb_card *)arg; +#endif + struct avt_pdu_aaf_pcm_hdr *hdr = + (struct avt_pdu_aaf_pcm_hdr *)&avb_card->sd + .tx_buf[sizeof(struct ethhdr)]; +#ifndef AVB_USE_HIGH_RES_TIMER + unsigned long int numJiffies = + ((jiffies > avb_card->tx.last_timer_ts) ? + (jiffies - avb_card->tx.last_timer_ts) : + (1)); + snd_pcm_uframes_t frameCount = ((avb_card->tx.sr * numJiffies) / HZ); +#endif + + avtp_max_frames_per_packet = + ((ETH_DATA_LEN - sizeof(struct avt_pdu_aaf_pcm_hdr)) / + avb_card->tx.frame_size); + +#ifdef AVB_USE_HIGH_RES_TIMER + avb_card->tx.accum_frame_count += avtp_max_frames_per_packet; + avtp_frames_per_packet = avtp_max_frames_per_packet; + kt = ktime_set(0, avb_card->tx.timer_val); + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avtp_timer mfppk: %lu, fppk: %lu, frSz: %lu, sr: %lu, time: %lu", + avtp_max_frames_per_packet, avtp_frames_per_packet, + avb_card->tx.frame_size, avb_card->tx.sr, + avb_card->tx.timer_val); +#else + avb_card->tx.accum_frame_count += frameCount; + avtp_frames_per_packet = (avb_card->tx.sr / HZ); + avtp_frames_per_packet = + ((avtp_frames_per_packet > avtp_max_frames_per_packet) ? + (avtp_max_frames_per_packet) : + (avtp_frames_per_packet)); + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avtp_timer ct: %lu, mfppk: %lu, fppk: %lu, frSz: %lu, sr: %lu, HZ: %lu", + frameCount, avtp_max_frames_per_packet, avtp_frames_per_packet, + avb_card->tx.frame_size, avb_card->tx.sr, HZ); +#endif + + while (((avb_card->tx.accum_frame_count >= avtp_frames_per_packet) || + (avb_card->tx.pending_tx_frames <= avtp_frames_per_packet)) && +#ifdef AVB_USE_HIGH_RES_TIMER + ((avb_card->tx.pending_tx_frames > 0) && (i < 1))) { +#else + ((avb_card->tx.pending_tx_frames > 0) && (frameCount > 0) && + (i < 32))) { +#endif + i++; /* Just as a failsafe to quit loop */ + avb_card->tx.seq_no++; + hdr->h.f.seq_no = avb_card->tx.seq_no; + + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_aaf_pcm_hdr); + frames_to_copy = ((avb_card->tx.pending_tx_frames > + avtp_frames_per_packet) ? + (avtp_frames_per_packet) : + (avb_card->tx.pending_tx_frames)); + bytes_to_copy = (frames_to_copy * avb_card->tx.frame_size); + + bytes_avai = ((avb_card->tx.frame_count - avb_card->tx.hw_idx) * + avb_card->tx.frame_size); + bytes_avai = ((bytes_avai >= bytes_to_copy) ? (bytes_to_copy) : + (bytes_avai)); + + memcpy(&avb_card->sd.tx_buf[tx_size], + &avb_card->tx.tmp_buf[(avb_card->tx.hw_idx * + avb_card->tx.frame_size)], + bytes_avai); + + if (bytes_avai < bytes_to_copy) { + memcpy(&avb_card->sd.tx_buf[tx_size + bytes_avai], + &avb_card->tx.tmp_buf[0], + (bytes_to_copy - bytes_avai)); + } + + hdr->h.f.avtp_ts = avb_device.tx_ts[( + (avb_card->tx.hwnw_idx / avb_card->tx.period_size) % + AVB_MAX_TS_SLOTS)]; + hdr->h.f.stream_data_len = bytes_to_copy; + tx_size += bytes_to_copy; + + avb_card->sd.tx_iov.iov_base = avb_card->sd.tx_buf; + avb_card->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avb_card->sd.tx_msg_hdr.msg_iter, WRITE, + &avb_card->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avb_card->sd.sock, + &avb_card->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_avtp_timer Socket transmission fails %d \n", + err); + goto end; + } + + avb_card->tx.accum_frame_count = + ((avb_card->tx.accum_frame_count > frames_to_copy) ? + (avb_card->tx.accum_frame_count - + frames_to_copy) : + (0)); + + avb_card->tx.hw_idx += frames_to_copy; + avb_card->tx.hwnw_idx += frames_to_copy; + avb_card->tx.fill_size += frames_to_copy; + avb_card->tx.hw_idx = + ((avb_card->tx.hw_idx < avb_card->tx.frame_count) ? + (avb_card->tx.hw_idx) : + (avb_card->tx.hw_idx % + avb_card->tx.frame_count)); + avb_card->tx.pending_tx_frames -= frames_to_copy; + + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avtp_timer seq_no:%d, hw_idx: %lu, afrCt: %lu, penFrs:%lu, filSz:%lu", + hdr->h.f.seq_no, avb_card->tx.hw_idx, + avb_card->tx.accum_frame_count, + avb_card->tx.pending_tx_frames, avb_card->tx.fill_size); + + if (avb_card->tx.fill_size >= avb_card->tx.period_size) { + avb_card->tx.fill_size %= avb_card->tx.period_size; + snd_pcm_period_elapsed(avb_card->tx.substream); + } + } + +end: +#ifdef AVB_USE_HIGH_RES_TIMER + if (avb_card->tx.st == 1) { + hrtimer_forward_now(t, kt); + hr_res = HRTIMER_RESTART; + } + + return hr_res; +#else + if ((avb_card->tx.pending_tx_frames > 0) && + (!timer_pending(&avb_device.txTimer))) { + avb_device.txTimer.expires = jiffies + 1; + add_timer(&avb_device.txTimer); + } + avb_card->tx.last_timer_ts = jiffies; +#endif +} + +static int avb_capture_open(struct snd_pcm_substream *substream) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_capture_open"); + + substream->runtime->hw = avb_capture_hw; + + return 0; +} + +static int avb_capture_close(struct snd_pcm_substream *substream) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_capture_close"); + + return 0; +} + +static int avb_capture_hw_params(struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *hw_params) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_card->rx.substream = substream; + avb_card->rx.hw_idx = 0; + avb_card->rx.seq_no = 0; + avb_card->rx.hwnw_idx = 0; + avb_card->rx.fill_size = 0; + avb_card->rx.prev_hw_idx = 0; + avb_card->rx.socket_count = 0; + avb_card->rx.num_bytes_consumed = 0; + avb_card->rx.period_size = params_period_size(hw_params); + avb_card->rx.buffer_size = params_buffer_bytes(hw_params); + avb_card->rx.frame_count = params_buffer_size(hw_params); + avb_card->rx.frame_size = + params_buffer_bytes(hw_params) / params_buffer_size(hw_params); + + avb_log(AVB_KERN_NOT, + KERN_NOTICE + "avb_capture_hw_params buffer_size:%lu frame_size:%lu", + avb_card->rx.buffer_size, avb_card->rx.frame_size); + + memset(&avb_device.rx_ts[0], 0, (sizeof(int) * AVB_MAX_TS_SLOTS)); + avb_device.rx_idx = 0; + + avb_device.avtpwd = avb_init_and_queue_work(AVB_DELAY_WORK_AVTP, + (void *)avb_card, 1); + + avb_card->rx.tmp_buf = kmalloc(avb_card->rx.buffer_size, GFP_KERNEL); + + return 0; +} + +static int avb_capture_hw_free(struct snd_pcm_substream *substream) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_capture_hw_free"); + + if (avb_device.avtpwd != NULL) { + cancel_delayed_work((struct delayed_work *)avb_device.avtpwd); + kfree(avb_device.avtpwd); + avb_device.avtpwd = NULL; + } + + kfree(avb_card->rx.tmp_buf); + + return 0; +} + +static int avb_capture_prepare(struct snd_pcm_substream *substream) +{ + return 0; +} + +static int avb_capture_trigger(struct snd_pcm_substream *substream, int cmd) +{ + int ret = 0; + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + switch (cmd) { + case SNDRV_PCM_TRIGGER_START: + avb_log(AVB_KERN_NOT, + KERN_NOTICE "avb_capture_trigger: Start @ %lu", + jiffies); + avb_card->rx.start_ts = jiffies; + break; + case SNDRV_PCM_TRIGGER_STOP: + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_capture_trigger: Stop"); + break; + default: + avb_log(AVB_KERN_WARN, + KERN_WARNING "avb_capture_trigger: Unknown"); + ret = -EINVAL; + } + + return ret; +} + +static snd_pcm_uframes_t +avb_capture_pointer(struct snd_pcm_substream *substream) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_capture_pointer hw_idx:%lu numBytes:%lu", + avb_card->rx.hw_idx, avb_card->rx.num_bytes_consumed); + + return avb_card->rx.hw_idx; +} + +static int avb_capture_copy(struct snd_pcm_substream *substream, int channel, + unsigned long pos, void __user *dst, + unsigned long count) +{ + char *src_buf; + int copyres = 0; + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + + pos = bytes_to_frames(substream->runtime, pos); + count = bytes_to_frames(substream->runtime, count); + + src_buf = (char *)&avb_card->rx.tmp_buf[pos * avb_card->rx.frame_size]; + + copyres = copy_to_user(dst, src_buf, (count * avb_card->rx.frame_size)); + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_capture_copy: ch:%d, pos: %ld, ct: %ld, res: %d", + channel, pos, count, copyres); + + // TODO: Check if frames to bytes is needed + avb_card->rx.num_bytes_consumed += frames_to_bytes( + substream->runtime, count * avb_card->rx.frame_size); + + return count; +} + +static int avb_avtp_listen(struct avb_card *avb_card) +{ + int err = 0; + char *src_buf; + char *dest_buf; + int rx_off = 0; + int rx_size = 0; + int nrx_size = 0; + int avai_size = 0; + int rx_frames = 0; + int nrx_frames = 0; + int next_seq_no = 0; + int skipped_packets = 0; + mm_segment_t oldfs; + snd_pcm_uframes_t hw_idx = 0; + struct kvec vec; + struct avt_pdu_aaf_pcm_hdr *hdr = + (struct avt_pdu_aaf_pcm_hdr *)&avb_card->sd + .rx_buf[sizeof(struct ethhdr)]; + + memset(avb_card->sd.rx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + avb_card->sd.rx_iov.iov_base = avb_card->sd.rx_buf; + avb_card->sd.rx_iov.iov_len = AVB_MAX_ETH_FRAME_SIZE; + iov_iter_init(&avb_card->sd.rx_msg_hdr.msg_iter, READ, + &avb_card->sd.rx_iov, 1, AVB_MAX_ETH_FRAME_SIZE); + + vec.iov_len = AVB_MAX_ETH_FRAME_SIZE; + vec.iov_base = avb_card->sd.rx_buf; + + oldfs = get_fs(); + set_fs(KERNEL_DS); + err = kernel_recvmsg(avb_card->sd.sock, &avb_card->sd.rx_msg_hdr, &vec, + 1, AVB_MAX_ETH_FRAME_SIZE, MSG_DONTWAIT); + set_fs(oldfs); + + if (err > 0) { + avb_card->rx.socket_count++; + + rx_off = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_aaf_pcm_hdr); + src_buf = (char *)&avb_card->sd.rx_buf[rx_off]; + + rx_size = hdr->h.f.stream_data_len; + rx_frames = (rx_size / avb_card->rx.frame_size); + + avb_device.rx_ts[( + (avb_card->rx.hwnw_idx / avb_card->rx.period_size) % + AVB_MAX_TS_SLOTS)] = hdr->h.f.avtp_ts; + + next_seq_no = (avb_card->rx.seq_no + 1) % 256; + + if (next_seq_no != hdr->h.f.seq_no) { + avb_log(AVB_KERN_INFO, + KERN_ERR + "avb_listen missing frames from %d to %d \n", + avb_card->rx.seq_no, hdr->h.f.seq_no); + + skipped_packets = + ((hdr->h.f.seq_no >= avb_card->rx.seq_no) ? + (hdr->h.f.seq_no - + avb_card->rx.seq_no) : + ((hdr->h.f.seq_no + 255) - + avb_card->rx.seq_no)); + nrx_frames = (skipped_packets * rx_frames); + nrx_size = nrx_frames * avb_card->rx.frame_size; + + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_listen idx: %ld nrsz: %d, nrf: %d \n", + avb_card->rx.hw_idx, nrx_size, nrx_frames); + + avai_size = ((avb_card->rx.frame_count - + avb_card->rx.hw_idx) * + avb_card->rx.frame_size); + avai_size = ((avai_size < nrx_size) ? (avai_size) : + (nrx_size)); + dest_buf = (char *)&avb_card->rx + .tmp_buf[avb_card->rx.hw_idx * + avb_card->rx.frame_size]; + memset(dest_buf, 0, avai_size); + + if (avai_size < nrx_size) { + dest_buf = (char *)&avb_card->rx.tmp_buf[0]; + memset(dest_buf, 0, (nrx_size - avai_size)); + } + + hw_idx = ((avb_card->rx.hw_idx + nrx_frames) % + (avb_card->rx.frame_count)); + rx_frames = rx_frames + nrx_frames; + } else { + hw_idx = avb_card->rx.hw_idx; + } + + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_listen (%d) seq: %d, idx: %ld, sz: %d, ts: %u, rf: %d \n", + avb_card->rx.socket_count, hdr->h.f.seq_no, hw_idx, + rx_size, hdr->h.f.avtp_ts, rx_frames); + + avb_card->rx.seq_no = hdr->h.f.seq_no; + + avai_size = ((avb_card->rx.frame_count - hw_idx) * + avb_card->rx.frame_size); + avai_size = ((avai_size < rx_size) ? (avai_size) : (rx_size)); + dest_buf = (char *)&avb_card->rx + .tmp_buf[hw_idx * avb_card->rx.frame_size]; + memcpy(dest_buf, src_buf, avai_size); + + if (avai_size < rx_size) { + dest_buf = (char *)&avb_card->rx.tmp_buf[0]; + memcpy(dest_buf, &src_buf[avai_size], + (rx_size - avai_size)); + } + } else { + if (err != -11) + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_avtp_listen Socket reception fails %d \n", + err); + return 0; + } + + return rx_frames; +} + +static void avb_wq_fn(struct work_struct *work) +{ + int err = 0; + int rx_count = 0; + int fill_size = 0; + int rx_frames = -1; + int rx_loop_count = 0; + struct work_data *wd = (struct work_data *)work; + + if (wd->delayed_work_id == AVB_DELAY_WORK_MSRP) { + if (wd->dw.msrp->initialized == false) + wd->dw.msrp->initialized = avb_msrp_init(wd->dw.msrp); + + if (wd->dw.msrp->initialized == false) { + queue_delayed_work( + avb_device.wq, + (struct delayed_work *)avb_device.msrpwd, + 10000); + } else { + if (wd->dw.msrp->started == true) { + do { + err = avb_msrp_listen(wd->dw.msrp); + rx_count += ((err > 0) ? (1) : (0)); + } while (err > 0); + + if (rx_count == 0) { + if ((wd->dw.msrp->tx_state == + MSRP_DECLARATION_STATE_NONE) || + (wd->dw.msrp->tx_state == + MSRP_DECLARATION_STATE_READY)) { + avb_msrp_talkerdeclarations( + wd->dw.msrp, true, + wd->dw.msrp->tx_state); + } + } else { + if ((wd->dw.msrp->tx_state == + MSRP_DECLARATION_STATE_NONE) || + (wd->dw.msrp->tx_state == + MSRP_DECLARATION_STATE_UNKNOWN)) { + avb_msrp_talkerdeclarations( + wd->dw.msrp, true, + wd->dw.msrp->tx_state); + } + + if ((wd->dw.msrp->rx_state == + MSRP_DECLARATION_STATE_NONE) || + (wd->dw.msrp->rx_state == + MSRP_DECLARATION_STATE_READY)) { + avb_msrp_listenerdeclarations( + wd->dw.msrp, true, + wd->dw.msrp->rx_state); + } + + wd->dw.msrp->tx_state = + ((wd->dw.msrp->tx_state == + MSRP_DECLARATION_STATE_UNKNOWN) ? + (MSRP_DECLARATION_STATE_NONE) : + (wd->dw.msrp + ->tx_state)); + wd->dw.msrp->rx_state = + ((wd->dw.msrp->rx_state == + MSRP_DECLARATION_STATE_UNKNOWN) ? + (MSRP_DECLARATION_STATE_NONE) : + (wd->dw.msrp + ->rx_state)); + } + } + + avb_msrp_domaindeclarations(wd->dw.msrp); + + queue_delayed_work( + avb_device.wq, + (struct delayed_work *)avb_device.msrpwd, + msecs_to_jiffies(2000)); + } + } else if (wd->delayed_work_id == AVB_DELAY_WORK_AVDECC) { + if (wd->dw.avdecc->initialized == false) + wd->dw.avdecc->initialized = + avb_avdecc_init(wd->dw.avdecc); + + if (wd->dw.avdecc->initialized == false) { + queue_delayed_work( + avb_device.wq, + (struct delayed_work *)avb_device.avdeccwd, + 10000); + } else { + if (wd->dw.avdecc->last_ADP_adv_jiffy == 0) + avb_adp_discover(wd->dw.avdecc); + if (((jiffies - wd->dw.avdecc->last_ADP_adv_jiffy) >= + 2000) || + (wd->dw.avdecc->last_ADP_adv_jiffy == 0)) { + avb_adp_advertise(wd->dw.avdecc); + avb_maap_announce(wd->dw.avdecc); + wd->dw.avdecc->last_ADP_adv_jiffy = jiffies; + } + avb_avdecc_listen_and_respond(wd->dw.avdecc, + &avb_device.msrp); + queue_delayed_work( + avb_device.wq, + (struct delayed_work *)avb_device.avdeccwd, 1); + } + } else if (wd->delayed_work_id == AVB_DELAY_WORK_AVTP) { + memcpy(&avb_device.card, wd->dw.card, sizeof(struct avb_card)); + + do { + rx_frames = avb_avtp_listen(&avb_device.card); + + if (rx_frames > 0) { + avb_device.card.rx.hw_idx += rx_frames; + avb_device.card.rx.hwnw_idx += rx_frames; + avb_device.card.rx.hw_idx %= + avb_device.card.rx.frame_count; + + if (avb_device.card.rx.hw_idx < + avb_device.card.rx.prev_hw_idx) + fill_size = + avb_device.card.rx.frame_count + + avb_device.card.rx.prev_hw_idx - + avb_device.card.rx.hw_idx; + else + fill_size = + avb_device.card.rx.hw_idx - + avb_device.card.rx.prev_hw_idx; + + avb_device.card.rx.prev_hw_idx = + avb_device.card.rx.hw_idx; + avb_device.card.rx.fill_size += fill_size; + + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_wq_fn: AVTP-%lu @ %lu rxFrms:%d hw_idx:%lu filSz: %lu", + rx_loop_count++, jiffies, rx_frames, + avb_device.card.rx.hw_idx, + avb_device.card.rx.fill_size); + + if (avb_device.card.rx.fill_size >= + avb_device.card.rx.period_size) { + avb_device.card.rx.fill_size %= + avb_device.card.rx.period_size; + snd_pcm_period_elapsed( + avb_device.card.rx.substream); + } + } else { + break; + } + + memcpy(wd->dw.card, &avb_device.card, + sizeof(struct avb_card)); + + } while (rx_frames > 0); + + if (avb_device.avtpwd != NULL) { + queue_delayed_work( + avb_device.wq, + (struct delayed_work *)avb_device.avtpwd, 1); + } + } else { + avb_log(AVB_KERN_INFO, KERN_INFO "avb_wq_fn: Unknown: %d", + wd->delayed_work_id); + } +} + +static int avb_pcm_new(struct avb_card *avbc, int device, int substreams) +{ + struct snd_pcm *pcm; + int err; + + err = snd_pcm_new(avbc->card, "AVB PCM", device, substreams, substreams, + &pcm); + if (err < 0) + return err; + + snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &avb_playback_ops); + snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &avb_capture_ops); + + pcm->private_data = avbc; + pcm->info_flags = 0; + + strcpy(pcm->name, "AVB PCM"); + + avbc->pcm[device] = pcm; + + return 0; +} + +static int avb_hwdep_open(struct snd_hwdep *hw, struct file *file) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_hwdep_open"); + + return 0; +} + +static int avb_hwdep_ioctl(struct snd_hwdep *hw, struct file *file, + unsigned int cmd, unsigned long arg) +{ + int res = 0; + + if (cmd == 0) { + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_hwdep_ioctl set ts: %ld @ idx: %d", arg, + avb_device.tx_idx); + avb_device.tx_ts[avb_device.tx_idx] = arg; + avb_device.tx_idx++; + avb_device.tx_idx %= AVB_MAX_TS_SLOTS; + } else { + res = copy_to_user((void *)arg, + &avb_device.rx_ts[avb_device.rx_idx], + sizeof(unsigned long)); + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_hwdep_ioctl get ts: %d @ %d, res: %d", + avb_device.rx_ts[avb_device.rx_idx], avb_device.rx_idx, + res); + avb_device.rx_idx++; + avb_device.rx_idx %= AVB_MAX_TS_SLOTS; + } + + return 0; +} + +static int avb_hwdep_release(struct snd_hwdep *hw, struct file *file) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_hwdep_release"); + + return 0; +} + +#ifdef CONFIG_PM_SLEEP + +static int avb_suspend(struct device *pdev) +{ + avb_log(AVB_KERN_INFO, KERN_INFO "avb_suspend"); + return 0; +} + +static int avb_resume(struct device *pdev) +{ + avb_log(AVB_KERN_INFO, KERN_INFO "avb_resume"); + return 0; +} + +#endif + +static int avb_probe(struct platform_device *devptr) +{ + struct snd_card *card; + struct avb_card *avb_card; + int dev = devptr->id; + int err; + + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_probe"); + + err = snd_card_new(&devptr->dev, index[dev], id[dev], THIS_MODULE, + sizeof(struct avb_card), &card); + + if (err < 0) { + avb_log(AVB_KERN_ERR, KERN_ERR "avb_probe card new err: %d", + err); + return err; + } + + avb_card = card->private_data; + avb_card->card = card; + + err = avb_pcm_new(avb_card, 0, pcm_substreams[dev]); + if (err < 0) { + avb_log(AVB_KERN_ERR, KERN_ERR "avb_probe card pcm new err: %d", + err); + goto __nodev; + } + + err = snd_hwdep_new(card, "avbhw", 0, &avb_device.hwdep); + if (err < 0) { + avb_log(AVB_KERN_ERR, + KERN_ERR "avb_probe card hwdep new err: %d", err); + goto __nodev; + } + + avb_device.hwdep->ops.open = avb_hwdep_open; + avb_device.hwdep->ops.ioctl = avb_hwdep_ioctl; + avb_device.hwdep->ops.release = avb_hwdep_release; + + strcpy(card->driver, "avb"); + strcpy(card->shortname, "avb"); + sprintf(card->longname, "avb %i", dev + 1); + err = snd_card_register(card); + if (!err) { + platform_set_drvdata(devptr, card); + + avb_card->sd.type = ETH_P_TSN; + avb_card->sd.destmac[0] = 0x01; + avb_card->sd.destmac[1] = 0x80; + avb_card->sd.destmac[2] = 0xC2; + avb_card->sd.destmac[3] = 0x00; + avb_card->sd.destmac[4] = 0x00; + avb_card->sd.destmac[5] = 0x0E; + + if (!avb_socket_init(&avb_card->sd, 100)) { + avb_log(AVB_KERN_ERR, + KERN_ERR "avb_probe socket init failed"); + err = -1; + goto __nodev; + } + + return 0; + } + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_probe card reg err: %d", err); + +__nodev: + snd_card_free(card); + + return err; +} + +static int avb_remove(struct platform_device *devptr) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_remove"); + snd_card_free(platform_get_drvdata(devptr)); + return 0; +} + +static void avb_remove_all(void) +{ + int i = 0; + + avb_log(AVB_KERN_NOT, KERN_NOTICE "avb_remove_all"); + + for (i = 0; i < numcards; i++) + platform_device_unregister(avbdevices[i]); +} + +static int __init alsa_avb_init(void) +{ + int i, err; + struct platform_device *dev; + avb_log(AVB_KERN_NOT, KERN_NOTICE "alsa_avb_init"); + + err = platform_driver_register(&avb_driver); + if (err < 0) { + avb_log(AVB_KERN_ERR, KERN_ERR "alsa_avb_init reg err %d", err); + return err; + } + + for (i = 0; i < SND_AVB_NUM_CARDS; i++) { + if (!enable[i]) + continue; + + dev = platform_device_register_simple(SND_AVB_DRIVER, i, NULL, + 0); + + if (IS_ERR(dev)) { + avb_log(AVB_KERN_ERR, + KERN_ERR "alsa_avb_init regsimple err"); + continue; + } + + if (!platform_get_drvdata(dev)) { + avb_log(AVB_KERN_ERR, + KERN_ERR "alsa_avb_init getdrvdata err"); + platform_device_unregister(dev); + continue; + } + + avbdevices[i] = dev; + numcards++; + } + + if (!numcards) { + avb_remove_all(); + } else { + memset(&avb_device, 0, sizeof(struct avb_device)); + + avb_device.wq = create_workqueue(AVB_WQ); + if (avb_device.wq == NULL) { + avb_log(AVB_KERN_ERR, KERN_ERR + "alsa_avb_init workqueue creation failed"); + return -1; + } + + avb_device.msrpwd = avb_init_and_queue_work( + AVB_DELAY_WORK_MSRP, (void *)&avb_device.msrp, 100); + avb_device.avdeccwd = avb_init_and_queue_work( + AVB_DELAY_WORK_AVDECC, (void *)&avb_device.avdecc, 100); + + avb_log(AVB_KERN_NOT, + KERN_NOTICE "alsa_avb_init done err: %d, numcards: %d", + err, numcards); + } + + return 0; +} + +static void __exit alsa_avb_exit(void) +{ + avb_log(AVB_KERN_NOT, KERN_NOTICE "alsa_avb_exit"); + + if (avb_device.msrpwd != NULL) { + cancel_delayed_work((struct delayed_work *)avb_device.msrpwd); + kfree(avb_device.msrpwd); + avb_device.msrpwd = NULL; + } + + if (avb_device.avdeccwd != NULL) { + cancel_delayed_work((struct delayed_work *)avb_device.avdeccwd); + kfree(avb_device.avdeccwd); + avb_device.avdeccwd = NULL; + } + + if (avb_device.wq != NULL) { + flush_workqueue(avb_device.wq); + destroy_workqueue(avb_device.wq); + } + + avb_remove_all(); + + platform_driver_unregister(&avb_driver); + + avb_log(AVB_KERN_NOT, KERN_NOTICE "alsa_avb_exit done"); +} + +module_init(alsa_avb_init) +module_exit(alsa_avb_exit) \ No newline at end of file diff --git a/sound/drivers/avb/avb.h b/sound/drivers/avb/avb.h new file mode 100644 index 0000000000000..fb0d778b3862c --- /dev/null +++ b/sound/drivers/avb/avb.h @@ -0,0 +1,39 @@ +#ifndef AVB_H +#define AVB_H + +#include "msrp.h" +#include "avdecc.h" +#include "avtp.h" + +struct work_data { + struct delayed_work work; + int delayed_work_id; + union delayed_work_data { + void *data; + struct msrp *msrp; + struct avb_card *card; + struct avdecc *avdecc; + } dw; +}; + +struct avb_device { + int tx_ts[AVB_MAX_TS_SLOTS]; + int rx_ts[AVB_MAX_TS_SLOTS]; + int tx_idx; + int rx_idx; + struct msrp msrp; + struct avdecc avdecc; + struct avb_card card; + struct snd_hwdep *hwdep; +#ifdef AVB_USE_HIGH_RES_TIMER + struct avb_hrtimer txTimer; +#else + struct timer_list txTimer; +#endif + struct work_data *avdeccwd; + struct work_data *msrpwd; + struct work_data *avtpwd; + struct workqueue_struct *wq; +}; + +#endif \ No newline at end of file diff --git a/sound/drivers/avb/avb_util.c b/sound/drivers/avb/avb_util.c new file mode 100644 index 0000000000000..1047dbd0b0227 --- /dev/null +++ b/sound/drivers/avb/avb_util.c @@ -0,0 +1,135 @@ +#include "avb_util.h" + +void avb_log(int level, char *fmt, ...) +{ + va_list args; + va_start(args, fmt); + + switch (level) { + case AVB_KERN_EMERG: + vprintk(fmt, args); + break; + case AVB_KERN_ALERT: + vprintk(fmt, args); + break; + case AVB_KERN_CRIT: + vprintk(fmt, args); + break; + case AVB_KERN_ERR: + vprintk(fmt, args); + break; + case AVB_KERN_WARN: + vprintk(fmt, args); + break; + case AVB_KERN_NOT: + vprintk(fmt, args); + break; +#ifdef AVB_DEBUG + case AVB_KERN_INFO: + vprintk(fmt, args); + break; + case AVB_KERN_DEBUG: + vprintk(fmt, args); + break; +#else + default: + break; +#endif + } + + va_end(args); +} + +bool avb_socket_init(struct socketdata *sd, int rx_timeout) +{ + int err = 0; + struct net_device *dev = NULL; + struct timeval ts_opts; + ts_opts.tv_sec = (rx_timeout / 1000); + ts_opts.tv_usec = (rx_timeout % 1000); + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_socket_init"); + + dev = dev_get_by_name_rcu(&init_net, "eth0"); + + if ((err = sock_create_kern(&init_net, AF_PACKET, SOCK_RAW, + htons(sd->type), &sd->sock)) != 0) { + avb_log(AVB_KERN_ERR, + KERN_ERR "avb_socket_init Socket creation fails %d \n", + err); + return false; + } + + memcpy(&sd->srcmac[0], dev->dev_addr, 6); + sd->if_idx = dev->ifindex; + + rtnl_lock(); + dev_set_promiscuity(dev, 1); + rtnl_unlock(); + + if ((err = kernel_setsockopt(sd->sock, SOL_SOCKET, SO_RCVTIMEO_OLD, + (void *)&ts_opts, sizeof(ts_opts))) != 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING "avb_msrp_init set rx timeout fails %d\n", + err); + return false; + } + + /* Index of the network device */ + sd->tx_sock_address.sll_family = AF_PACKET; + sd->tx_sock_address.sll_protocol = htons(sd->type); + sd->tx_sock_address.sll_ifindex = sd->if_idx; + /* Address length*/ + sd->tx_sock_address.sll_halen = ETH_ALEN; + /* Destination MAC */ + sd->tx_sock_address.sll_addr[0] = sd->destmac[0]; + sd->tx_sock_address.sll_addr[1] = sd->destmac[1]; + sd->tx_sock_address.sll_addr[2] = sd->destmac[2]; + sd->tx_sock_address.sll_addr[3] = sd->destmac[3]; + sd->tx_sock_address.sll_addr[4] = sd->destmac[4]; + sd->tx_sock_address.sll_addr[5] = sd->destmac[5]; + + /* Set the message header */ + sd->tx_msg_hdr.msg_control = NULL; + sd->tx_msg_hdr.msg_controllen = 0; + sd->tx_msg_hdr.msg_flags = 0; + sd->tx_msg_hdr.msg_name = &sd->tx_sock_address; + sd->tx_msg_hdr.msg_namelen = sizeof(struct sockaddr_ll); + sd->tx_msg_hdr.msg_iocb = NULL; + + /* Index of the network device */ + sd->rx_sock_address.sll_family = AF_PACKET; + sd->rx_sock_address.sll_protocol = htons(sd->type); + sd->rx_sock_address.sll_ifindex = sd->if_idx; + /* Address length*/ + sd->rx_sock_address.sll_halen = ETH_ALEN; + /* Destination MAC */ + sd->rx_sock_address.sll_addr[0] = sd->destmac[0]; + sd->rx_sock_address.sll_addr[1] = sd->destmac[1]; + sd->rx_sock_address.sll_addr[2] = sd->destmac[2]; + sd->rx_sock_address.sll_addr[3] = sd->destmac[3]; + sd->rx_sock_address.sll_addr[4] = sd->destmac[4]; + sd->rx_sock_address.sll_addr[5] = sd->destmac[5]; + + /* Set the message header */ + sd->rx_msg_hdr.msg_control = NULL; + sd->rx_msg_hdr.msg_controllen = 0; + sd->rx_msg_hdr.msg_flags = 0; + sd->rx_msg_hdr.msg_name = &sd->rx_sock_address; + sd->rx_msg_hdr.msg_namelen = sizeof(struct sockaddr_ll); + sd->rx_msg_hdr.msg_iocb = NULL; + sd->rx_iov.iov_base = sd->rx_buf; + sd->rx_iov.iov_len = AVB_MAX_ETH_FRAME_SIZE; + iov_iter_init(&sd->rx_msg_hdr.msg_iter, READ, &sd->rx_iov, 1, + AVB_MAX_ETH_FRAME_SIZE); + + if ((err = kernel_bind(sd->sock, + (struct sockaddr *)&sd->rx_sock_address, + sizeof(sd->rx_sock_address))) != 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING "avb socket binding fails %d\n", err); + return false; + } + + return true; +} \ No newline at end of file diff --git a/sound/drivers/avb/avb_util.h b/sound/drivers/avb/avb_util.h new file mode 100644 index 0000000000000..331ab6ca89749 --- /dev/null +++ b/sound/drivers/avb/avb_util.h @@ -0,0 +1,307 @@ +#ifndef AVB_UTIL_H +#define AVB_UTIL_H + +#include +#include +#include +#include +#include +#include + +#define SND_AVB_DRIVER "snd_avb" + +#define AVB_DEBUG + +#define ETH_MSRP (0x22EA) + +#define SND_AVB_NUM_CARDS (SNDRV_CARDS) + +#ifdef CONFIG_PM_SLEEP +#define AVB_PM_OPS &avb_pm +#else +#define AVB_PM_OPS NULL +#endif + +#define AVB_USE_HIGH_RES_TIMER + +#define AVB_KERN_EMERG 0 /* system is unusable */ +#define AVB_KERN_ALERT 1 /* action must be taken immediately */ +#define AVB_KERN_CRIT 2 /* critical conditions */ +#define AVB_KERN_ERR 3 /* error conditions */ +#define AVB_KERN_WARN 4 /* warning conditions */ +#define AVB_KERN_NOT 5 /* normal but significant condition */ +#define AVB_KERN_INFO 6 /* informational */ +#define AVB_KERN_DEBUG 7 /* debug-level messages */ + +#define AVB_WQ "AVBWQ" + +#define AVB_DELAY_WORK_MSRP (0) +#define AVB_DELAY_WORK_AVTP (1) +#define AVB_DELAY_WORK_AVDECC (2) + +#define AVB_MAX_TS_SLOTS (12) +#define AVB_AVTP_AAF_SAMPLES_PER_PACKET \ + (192) /* 4ms * 48KHz i.e. Maxframes per jiffy for HZ=250 */ +#define AVB_MSRP_ETH_FRAME_SIZE (2048) +#define AVB_MAX_ETH_FRAME_SIZE (AVB_MSRP_ETH_FRAME_SIZE) + +#define MSRP_ATTRIBUTE_TYPE_TALKER_ADVERTISE_VECTOR (1) +#define MSRP_ATTRIBUTE_TYPE_TALKER_FAILED_VECTOR (2) +#define MSRP_ATTRIBUTE_TYPE_LISTENER_VECTOR (3) +#define MSRP_ATTRIBUTE_TYPE_DOMAIN_VECTOR (4) + +#define MSRP_ATTRIBUTE_LEN_TALKER_ADVERTISE_VECTOR (25) +#define MSRP_ATTRIBUTE_LEN_TALKER_FAILED_VECTOR (34) +#define MSRP_ATTRIBUTE_LEN_LISTENER_VECTOR (8) +#define MSRP_ATTRIBUTE_LEN_DOMAIN_VECTOR (4) + +#define MSRP_MAX_FRAME_SIZE_48KHZ_AUDIO (80) + +#define MSRP_MAX_INTERVAL_FRAME_48KHZ_AUDIO (1) + +#define MSRP_ATTRIBUTE_EVENT_NEW (0) +#define MSRP_ATTRIBUTE_EVENT_JOININ (1) +#define MSRP_ATTRIBUTE_EVENT_IN (2) +#define MSRP_ATTRIBUTE_EVENT_JOINMT (3) +#define MSRP_ATTRIBUTE_EVENT_MT (4) +#define MSRP_ATTRIBUTE_EVENT_LEAVE (5) + +#define MSRP_DECLARATION_STATE_NONE (0) +#define MSRP_DECLARATION_STATE_IGNORE (0) +#define MSRP_DECLARATION_STATE_ASKING_FAILED (1) +#define MSRP_DECLARATION_STATE_READY (2) +#define MSRP_DECLARATION_STATE_READY_FAILED (3) +#define MSRP_DECLARATION_STATE_UNKNOWN (255) + +#define MSRP_THREE_PACK(a, b, c) (u8)((((a * 6) + b) * 6) + c) +#define MSRP_THREE_PACK_GET_A(x) (u8)(x / 36) +#define MSRP_THREE_PACK_GET_B(x) (u8)((x / 6) % 6) +#define MSRP_THREE_PACK_GET_C(x) (u8)(x % 6) +#define MSRP_FOUR_PACK(a, b, c, d) (u8)((a * 64) + (b * 16) + (c * 4) + (d)) +#define MSRP_FOUR_PACK_GET_A(x) (u8)(x / 64) + +#define AVTP_PDU_COMMON_STREAM_HEADER_LENGTH (24) +#define AVTP_PDU_COMMON_CONTROL_HEADER_LENGTH (12) + +#define AVB_AVTP_AAF_VERSION (0) + +#define AVB_AVTP_SUBTYPE_AAF (2) +#define AVB_AVTP_SUBTYPE_ADP (0xFA) +#define AVB_AVTP_SUBTYPE_AECP (0xFB) +#define AVB_AVTP_SUBTYPE_ACMP (0xFC) +#define AVB_AVTP_SUBTYPE_MAAP (0xFE) + +#define AVB_AVTP_AAF_FORMAT_USER_SP (0) +#define AVB_AVTP_AAF_FORMAT_32_BIT_FLOAT (1) +#define AVB_AVTP_AAF_FORMAT_32_BIT_INT (2) +#define AVB_AVTP_AAF_FORMAT_24_bit_INT (3) +#define AVB_AVTP_AAF_FORMAT_16_BIT_INT (4) +#define AVB_AVTP_AAF_FORMAT_32_BIT_AES3 (5) + +#define AVB_AVTP_AAF_NSR_USER_SP (0x0) +#define AVB_AVTP_AAF_NSR_8_KHZ (0x1) +#define AVB_AVTP_AAF_NSR_16_KHZ (0x2) +#define AVB_AVTP_AAF_NSR_32_KHZ (0x3) +#define AVB_AVTP_AAF_NSR_44_1_KHZ (0x4) +#define AVB_AVTP_AAF_NSR_48_KHZ (0x5) +#define AVB_AVTP_AAF_NSR_88_2_KHZ (0x6) +#define AVB_AVTP_AAF_NSR_96_KHZ (0x7) +#define AVB_AVTP_AAF_NSR_176_4_KHZ (0x8) +#define AVB_AVTP_AAF_NSR_192_KHZ (0x9) +#define AVB_AVTP_AAF_NSR_24_KHZ (0xA) + +#define AVB_ADP_MSGTYPE_ENTITY_AVAILABLE (0x0) +#define AVB_ADP_MSGTYPE_ENTITY_DEPARTING (0x1) +#define AVB_ADP_MSGTYPE_ENTITY_DISCOVER (0x2) + +#define AVB_ADP_CONTROL_DATA_LENGTH (56) + +#define AVB_AECP_MSGTYPE_AEM_COMMAND (0x00) +#define AVB_AECP_MSGTYPE_AEM_RESPONSE (0x01) + +#define AVB_AEM_CMD_ENTITY_ACQUIRE (0x00) +#define AVB_AEM_CMD_ENTITY_LOCK (0x01) +#define AVB_AEM_CMD_ENTITY_AVAILABLE (0x02) +#define AVB_AEM_CMD_CTRL_AVAILABLE (0x03) +#define AVB_AEM_CMD_READ_DESCP (0x04) +#define AVB_AEM_CMD_WRITE_DESCP (0x05) +#define AVB_AEM_CMD_SET_CONFIG (0x06) +#define AVB_AEM_CMD_GET_CONFIG (0x07) +#define AVB_AEM_CMD_SET_STREAM_FORMAT (0x08) +#define AVB_AEM_CMD_SET_STREAM_INFO (0x0e) +#define AVB_AEM_CMD_GET_STREAM_INFO (0x0f) +#define AVB_AEM_CMD_REGISTER_UNSOLICITED_NOTIFICATION (0x24) +#define AVB_AEM_CMD_GET_COUNTERS (0x029) + +#define AVB_AEM_DESCP_ENTITY (0x00) +#define AVB_AEM_DESCP_CONFIGURATION (0x01) +#define AVB_AEM_DESCP_AUDIO_UNIT (0x02) +#define AVB_AEM_DESCP_STREAM_IP (0x05) +#define AVB_AEM_DESCP_STREAM_OP (0x06) +#define AVB_AEM_DESCP_JACK_IP (0x07) +#define AVB_AEM_DESCP_JACK_OP (0x08) +#define AVB_AEM_DESCP_AVBINTERFACE (0x09) +#define AVB_AEM_DESCP_CLOCKSOURCE (0x0a) +#define AVB_AEM_DESCP_LOCALE (0x0c) +#define AVB_AEM_DESCP_STRINGS (0x0d) +#define AVB_AEM_DESCP_STREAM_PORT_IP (0x0e) +#define AVB_AEM_DESCP_STREAM_PORT_OP (0x0f) +#define AVB_AEM_DESCP_EXT_PORT_IP (0x10) +#define AVB_AEM_DESCP_EXT_PORT_OP (0x11) +#define AVB_AEM_DESCP_AUDIO_CLUSTER (0x14) +#define AVB_AEM_DESCP_AUDIO_MAP (0x17) +#define AVB_AEM_DESCP_CLOCK_DOMAIN (0x24) +#define AVB_AEM_DESCP_INVALID (0xffff) + +#define AVB_AEM_RES_SUCCESS (0x00) +#define AVB_AEM_RES_NOT_IMPLEMENTED (0x01) +#define AVB_AEM_RES_NO_SUCH_DESCRIPTOR (0x02) + +#define AVB_AEM_MAX_DESCP_COUNT (7) + +#define AVB_AEM_STREAM_FORMAT_AVTP (0x02) +#define AVB_AEM_MAX_SUPP_FORMATS (6) + +#define AVB_ACMP_MSGTYPE_CONNECT_TX_CMD (0x00) +#define AVB_ACMP_MSGTYPE_CONNECT_TX_RESP (0x01) +#define AVB_ACMP_MSGTYPE_DISCONNECT_TX_CMD (0x02) +#define AVB_ACMP_MSGTYPE_DISCONNECT_TX_RESP (0x03) +#define AVB_ACMP_MSGTYPE_GET_TX_STATE_CMD (0x04) +#define AVB_ACMP_MSGTYPE_GET_TX_STATE_RESP (0x05) +#define AVB_ACMP_MSGTYPE_CONNECT_RX_CMD (0x06) +#define AVB_ACMP_MSGTYPE_CONNECT_RX_RESP (0x07) +#define AVB_ACMP_MSGTYPE_DISCONNECT_RX_CMD (0x08) +#define AVB_ACMP_MSGTYPE_DISCONNECT_RX_RESP (0x09) +#define AVB_ACMP_MSGTYPE_GET_RX_STATE_CMD (0x0A) +#define AVB_ACMP_MSGTYPE_GET_RX_STATE_RESP (0x0B) +#define AVB_ACMP_MSGTYPE_GET_TX_CONN_CMD (0x0C) +#define AVB_ACMP_MSGTYPE_GET_TX_CONN_RESP (0x0D) + +#define AVB_ACMP_STATUS_SUCCESS (0x00) +#define AVB_ACMP_STATUS_LISTENER_UNKNOWN_ID (0x01) +#define AVB_ACMP_STATUS_TALKER_UNKNOWN_ID (0x02) +#define AVB_ACMP_STATUS_TALKER_DEST_MAC_FAIL (0x03) +#define AVB_ACMP_STATUS_TALKER_NO_STREAM_IDX (0x04) +#define AVB_ACMP_STATUS_TALKER_NO_BANDWIDTH (0x05) +#define AVB_ACMP_STATUS_TALKER_EXCLUSIVE (0x06) +#define AVB_ACMP_STATUS_LISTENER_TALKER_TIMEOUT (0x07) +#define AVB_ACMP_STATUS_LISTENER_EXCLUSIVE (0x08) +#define AVB_ACMP_STATUS_STATE_UNAVAILABLE (0x09) +#define AVB_ACMP_STATUS_NOT_CONNECTED (0x0a) +#define AVB_ACMP_STATUS_NO_SUCH_CONNECTION (0x0b) +#define AVB_ACMP_STATUS_COULD_NOT_SEND_MESSAGE (0x0c) +#define AVB_ACMP_STATUS_TALKER_MISBEHAVING (0x0d) +#define AVB_ACMP_STATUS_LISTENER_MISBEHAVING (0x0e) +#define AVB_ACMP_STATUS_RFU (0x0f) +#define AVB_ACMP_STATUS_CONTROLLER_NOT_AUTHORIZED (0x10) +#define AVB_ACMP_STATUS_INCOMPATIBLE_REQUEST (0x11) +#define AVB_ACMP_STATUS_NOT_SUPPORTED (0x1f) + +#define AVB_AVTP_AAF_HDR_GET_SV(hdr) ((hdr->h.f.b1.sv & 0x80) >> 7) +#define AVB_AVTP_AAF_HDR_SET_SV(hdr, val) \ + (hdr->h.f.b1.sv = (hdr->h.f.b1.sv | ((val << 7) & 0x80))) +#define AVB_AVTP_AAF_HDR_GET_VER(hdr) ((hdr->h.f.b1.version & 0x70) >> 4) +#define AVB_AVTP_AAF_HDR_SET_VER(hdr, val) \ + (hdr->h.f.b1.version = (hdr->h.f.b1.version | ((val << 4) & 0x70))) +#define AVB_AVTP_AAF_HDR_GET_MR(hdr) ((hdr->h.f.b1.mr & 0x08) >> 3) +#define AVB_AVTP_AAF_HDR_SET_MR(hdr, val) \ + (hdr->h.f.b1.mr = (hdr->h.f.b1.mr | ((val << 3) & 0x08))) +#define AVB_AVTP_AAF_HDR_GET_TSV(hdr) (hdr->h.f.b1.ts_valid & 0x01) +#define AVB_AVTP_AAF_HDR_SET_TSV(hdr, val) \ + (hdr->h.f.b1.ts_valid = (hdr->h.f.b1.ts_valid | (val & 0x01))) +#define AVB_AVTP_AAF_HDR_GET_TU(hdr) (hdr->h.f.b2.tu & 0x01) +#define AVB_AVTP_AAF_HDR_SET_TU(hdr, val) \ + (hdr->h.f.b2.tu = (hdr->h.f.b2.tu | (val & 0x01))) +#define AVB_AVTP_AAF_HDR_GET_NSR(hdr) ((hdr->h.f.fsd1.nsr & 0xF0) >> 4) +#define AVB_AVTP_AAF_HDR_SET_NSR(hdr, val) \ + (hdr->h.f.fsd1.nsr = (hdr->h.f.fsd1.nsr | ((val << 4) & 0xF0))) +#define AVB_AVTP_AAF_HDR_GET_CPF(hdr) (hdr->h.f.fsd1.cpf & 0x03) +#define AVB_AVTP_AAF_HDR_SET_CPF(hdr, val) \ + (hdr->h.f.fsd1.cpf = (hdr->h.f.fsd1.cpf | (val & 0x03))) +#define AVB_AVTP_AAF_HDR_GET_SP(hdr) ((hdr->h.f.fsd2.sp & 0x10) >> 4) +#define AVB_AVTP_AAF_HDR_SET_SP(hdr, val) \ + (hdr->h.f.fsd2.sp = (hdr->h.f.fsd2.sp | ((val << 4) & 0x10))) +#define AVB_AVTP_AAF_HDR_GET_EVT(hdr) (hdr->h.f.fsd2.evt & 0x0F) +#define AVB_AVTP_AAF_HDR_SET_EVT(hdr, val) \ + (hdr->h.f.fsd2.evt = (hdr->h.f.fsd2.evt | (val & 0x0F))) + +#define AVB_AVTPDU_CTRL_HDR_GET_SV(hdr) ((hdr->h.f.b1.sv & 0x80) >> 7) +#define AVB_AVTPDU_CTRL_HDR_SET_SV(hdr, val) \ + (hdr->h.f.b1.sv = (hdr->h.f.b1.sv | ((val << 7) & 0x80))) +#define AVB_AVTPDU_CTRL_HDR_GET_VER(hdr) ((hdr->h.f.b1.version & 0x70) >> 4) +#define AVB_AVTPDU_CTRL_HDR_SET_VER(hdr, val) \ + (hdr->h.f.b1.version = (hdr->h.f.b1.version | ((val << 4) & 0x70))) +#define AVB_AVTPDU_CTRL_HDR_GET_MSGTYPE(hdr) ((hdr->h.f.b1.msg_type & 0x0f)) +#define AVB_AVTPDU_CTRL_HDR_SET_MSGTYPE(hdr, val) \ + (hdr->h.f.b1.msg_type = (hdr->h.f.b1.msg_type | ((val)&0x0f))) +#define AVB_AVTPDU_CTRL_HDR_GET_VALIDTIME(hdr) \ + ((hdr->h.f.b2.valid_time & 0xf8) >> 3) +#define AVB_AVTPDU_CTRL_HDR_SET_VALIDTIME(hdr, val) \ + (hdr->h.f.b2.valid_time = \ + (hdr->h.f.b2.valid_time | ((val << 3) & 0xf8))) +#define AVB_AVTPDU_CTRL_HDR_GET_DATALEN(hdr) \ + ((((hdr->h.f.b2.data_len & 0x0007) << 8) & 0xff00) | \ + (hdr->h.f.data_len & 0x00ff)) +#define AVB_AVTPDU_CTRL_HDR_SET_DATALEN(hdr, val) \ + ((hdr->h.f.b2.data_len = \ + (hdr->h.f.b2.data_len | ((val >> 8) & 0x0007))), \ + (hdr->h.f.data_len = (val & 0x00ff))) + +struct socketdata { + int type; + int if_idx; + char srcmac[6]; + char destmac[6]; + struct socket *sock; + struct iovec tx_iov; + struct iovec rx_iov; + struct msghdr tx_msg_hdr; + struct sockaddr_ll tx_sock_address; + struct msghdr rx_msg_hdr; + struct sockaddr_ll rx_sock_address; + char tx_buf[AVB_MAX_ETH_FRAME_SIZE]; + char rx_buf[AVB_MAX_ETH_FRAME_SIZE]; +}; + +struct stream_info { + int sr; + int st; + int seq_no; + int socket_count; + unsigned long int timer_val; + unsigned long int last_timer_ts; + unsigned long int start_ts; + snd_pcm_uframes_t hw_idx; + snd_pcm_uframes_t hwnw_idx; + snd_pcm_uframes_t num_bytes_consumed; + snd_pcm_uframes_t period_size; + snd_pcm_uframes_t pending_tx_frames; + snd_pcm_uframes_t frame_size; + snd_pcm_uframes_t buffer_size; + snd_pcm_uframes_t fill_size; + snd_pcm_uframes_t prev_hw_idx; + snd_pcm_uframes_t frame_count; + snd_pcm_uframes_t accum_frame_count; + struct snd_pcm_substream *substream; + unsigned char *tmp_buf; +}; + +struct avb_card { + struct socketdata sd; + struct snd_card *card; + struct snd_pcm *pcm[1]; + struct stream_info tx; + struct stream_info rx; +}; + +#ifdef AVB_USE_HIGH_RES_TIMER +struct avb_hrtimer { + struct hrtimer timer; + struct avb_card *card; +}; +#endif + +bool avb_socket_init(struct socketdata *sd, int rx_timeout); +void avb_log(int level, char *fmt, ...); + +#endif \ No newline at end of file diff --git a/sound/drivers/avb/avdecc.c b/sound/drivers/avb/avdecc.c new file mode 100644 index 0000000000000..3fdc038ad9f14 --- /dev/null +++ b/sound/drivers/avb/avdecc.c @@ -0,0 +1,1602 @@ +#include "avdecc.h" + +bool avb_avdecc_init(struct avdecc *avdecc) +{ + avb_log(AVB_KERN_INFO, KERN_INFO "avb_avdecc_init"); + + avdecc->sd.type = ETH_P_TSN; + avdecc->sd.destmac[0] = 0x91; + avdecc->sd.destmac[1] = 0xe0; + avdecc->sd.destmac[2] = 0xf0; + avdecc->sd.destmac[3] = 0x01; + avdecc->sd.destmac[4] = 0x00; + avdecc->sd.destmac[5] = 0x00; + + avdecc->acmp_tx_state = AVB_ACMP_STATUS_NOT_CONNECTED; + avdecc->acmp_rx_state = AVB_ACMP_STATUS_NOT_CONNECTED; + + return avb_socket_init(&avdecc->sd, 1000); +} + +static void avb_acdecc_init_and_fill_eth_hdr(struct avdecc *avdecc, + u8 multicast) +{ + struct ethhdr *eh = (struct ethhdr *)&avdecc->sd.tx_buf[0]; + struct ethhdr *reh = (struct ethhdr *)&avdecc->sd.rx_buf[0]; + + /* Initialize it */ + memset(avdecc->sd.tx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + + /* Fill in the Ethernet header */ + if (multicast == 0) { + eh->h_dest[0] = reh->h_source[0]; + eh->h_dest[1] = reh->h_source[1]; + eh->h_dest[2] = reh->h_source[2]; + eh->h_dest[3] = reh->h_source[3]; + eh->h_dest[4] = reh->h_source[4]; + eh->h_dest[5] = reh->h_source[5]; + } else { + eh->h_dest[0] = 0x91; + eh->h_dest[1] = 0xe0; + eh->h_dest[2] = 0xf0; + eh->h_dest[3] = 0x01; + eh->h_dest[4] = 0x00; + eh->h_dest[5] = 0x00; + } + eh->h_source[0] = avdecc->sd.srcmac[0]; + eh->h_source[1] = avdecc->sd.srcmac[1]; + eh->h_source[2] = avdecc->sd.srcmac[2]; + eh->h_source[3] = avdecc->sd.srcmac[3]; + eh->h_source[4] = avdecc->sd.srcmac[4]; + eh->h_source[5] = avdecc->sd.srcmac[5]; + + /* Fill in Ethertype field */ + eh->h_proto = htons(avdecc->sd.type); +} + +static void avb_acdecc_fill_AVTP_ctrl_hdr(struct avdecc *avdecc, u8 sub_type, + u8 msg_type, u8 status, u16 data_len, + int stream_id) +{ + struct avt_pdu_control_hdr *hdr = + (struct avt_pdu_control_hdr *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr)]; + + hdr->h.f.sub_type = sub_type; + AVB_AVTPDU_CTRL_HDR_SET_SV(hdr, 0); + AVB_AVTPDU_CTRL_HDR_SET_VER(hdr, 0); + AVB_AVTPDU_CTRL_HDR_SET_MSGTYPE(hdr, msg_type); + AVB_AVTPDU_CTRL_HDR_SET_VALIDTIME(hdr, status); + AVB_AVTPDU_CTRL_HDR_SET_DATALEN(hdr, data_len); + if (stream_id == 0) { + hdr->h.f.stream_id[0] = + avdecc->sd.srcmac[0]; //avdecc->sd.srcmac[0]; + hdr->h.f.stream_id[1] = + avdecc->sd.srcmac[1]; //avdecc->sd.srcmac[1]; + hdr->h.f.stream_id[2] = + avdecc->sd.srcmac[2]; //avdecc->sd.srcmac[2]; + hdr->h.f.stream_id[3] = avdecc->sd.srcmac[3]; //0xff; + hdr->h.f.stream_id[4] = avdecc->sd.srcmac[4]; //0xfe; + hdr->h.f.stream_id[5] = + avdecc->sd.srcmac[5]; //avdecc->sd.srcmac[3]; + hdr->h.f.stream_id[6] = 0; //avdecc->sd.srcmac[4]; + hdr->h.f.stream_id[7] = 1; //avdecc->sd.srcmac[5]; + } else if (stream_id > 0) { + hdr->h.f.stream_id[0] = avdecc->sd.srcmac[0]; + hdr->h.f.stream_id[1] = avdecc->sd.srcmac[1]; + hdr->h.f.stream_id[2] = avdecc->sd.srcmac[2]; + hdr->h.f.stream_id[3] = 0xff; + hdr->h.f.stream_id[4] = 0xfe; + hdr->h.f.stream_id[5] = avdecc->sd.srcmac[3]; + hdr->h.f.stream_id[6] = avdecc->sd.srcmac[4]; + hdr->h.f.stream_id[7] = avdecc->sd.srcmac[5]; + } else { + hdr->h.f.stream_id[0] = 0; + hdr->h.f.stream_id[1] = 0; + hdr->h.f.stream_id[2] = 0; + hdr->h.f.stream_id[3] = 0; + hdr->h.f.stream_id[4] = 0; + hdr->h.f.stream_id[5] = 0; + hdr->h.f.stream_id[6] = 0; + hdr->h.f.stream_id[7] = 0; + } +} + +void avb_maap_announce(struct avdecc *avdecc) +{ + int tx_size = 0; + int err = 0; + + struct maap_pdu *pdu = + (struct maap_pdu *)&avdecc->sd.tx_buf[sizeof(struct ethhdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_maap_announce"); + + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_MAAP, 3, 1, 16, + -1); + + pdu->req_MAC[0] = 0x91; + pdu->req_MAC[1] = 0xe0; + pdu->req_MAC[2] = 0xf0; + pdu->req_MAC[3] = 0x00; + pdu->req_MAC[4] = 0x33; + pdu->req_MAC[5] = 0x4b; + + pdu->req_count = 2; + + tx_size = sizeof(struct ethhdr) + sizeof(struct maap_pdu); + + avdecc->sd.tx_iov.iov_base = avdecc->sd.tx_buf; + avdecc->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avdecc->sd.tx_msg_hdr.msg_iter, WRITE, + &avdecc->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avdecc->sd.sock, &avdecc->sd.tx_msg_hdr)) <= + 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_maap_announce Socket transmission fails %d \n", + err); + return; + } +} + +void avb_adp_discover(struct avdecc *avdecc) +{ + int tx_size = 0; + int err = 0; + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_adp_discover"); + + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_ADP, + AVB_ADP_MSGTYPE_ENTITY_DISCOVER, 31, + AVB_ADP_CONTROL_DATA_LENGTH, 1); + + tx_size = sizeof(struct ethhdr) + sizeof(struct avt_pdu_control_hdr) + + AVB_ADP_CONTROL_DATA_LENGTH; + + avdecc->sd.tx_iov.iov_base = avdecc->sd.tx_buf; + avdecc->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avdecc->sd.tx_msg_hdr.msg_iter, WRITE, + &avdecc->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avdecc->sd.sock, &avdecc->sd.tx_msg_hdr)) <= + 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_adp_discover Socket transmission fails %d \n", + err); + return; + } +} + +void avb_adp_advertise(struct avdecc *avdecc) +{ + int tx_size = 0; + int err = 0; + + struct adpdu *adpdu = + (struct adpdu *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_adp_advertise"); + + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_ADP, + AVB_ADP_MSGTYPE_ENTITY_AVAILABLE, 31, + AVB_ADP_CONTROL_DATA_LENGTH, 1); + + adpdu->entity_model_id[0] = avdecc->sd.srcmac[0]; + adpdu->entity_model_id[1] = avdecc->sd.srcmac[1]; + adpdu->entity_model_id[2] = avdecc->sd.srcmac[2]; + adpdu->entity_model_id[3] = avdecc->sd.srcmac[3]; + adpdu->entity_model_id[4] = avdecc->sd.srcmac[4]; + adpdu->entity_model_id[5] = avdecc->sd.srcmac[5]; + adpdu->entity_model_id[6] = 0x00; + adpdu->entity_model_id[7] = 0x01; + adpdu->entity_caps = htonl(0x00008508); + adpdu->talker_stream_sources = htons(1); + adpdu->talker_caps = htons(0x4001); + adpdu->listener_stream_sinks = htons(1); + adpdu->listener_caps = htons(0x4001); + adpdu->control_caps = 0; + adpdu->avai_idx = htonl(avdecc->adp_avai_idx++); + adpdu->gptp_grand_master_id[0] = + /* 0x8d; */ 0xa8; //avdecc->sd.srcmac[0]; + adpdu->gptp_grand_master_id[1] = + /* 0x85; */ 0x60; //avdecc->sd.srcmac[1]; + adpdu->gptp_grand_master_id[2] = + /* 0x90; */ 0xb6; //avdecc->sd.srcmac[2]; + adpdu->gptp_grand_master_id[3] = /* 0x2c; */ 0xFF; + adpdu->gptp_grand_master_id[4] = /* 0xdf; */ 0xFE; + adpdu->gptp_grand_master_id[5] = + /* 0xe9; */ 0x14; //avdecc->sd.srcmac[3]; + adpdu->gptp_grand_master_id[6] = + /* 0x00; */ 0xa6; //avdecc->sd.srcmac[4]; + adpdu->gptp_grand_master_id[7] = + /* 0x00; */ 0x3f; //avdecc->sd.srcmac[5]; + adpdu->gptp_domain_number = 0; + adpdu->iden_ctrl_idx = 0; + adpdu->interface_idx = 0; + + tx_size = sizeof(struct ethhdr) + sizeof(struct avt_pdu_control_hdr) + + AVB_ADP_CONTROL_DATA_LENGTH; + + avdecc->sd.tx_iov.iov_base = avdecc->sd.tx_buf; + avdecc->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avdecc->sd.tx_msg_hdr.msg_iter, WRITE, + &avdecc->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avdecc->sd.sock, &avdecc->sd.tx_msg_hdr)) <= + 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_adp_advertise Socket transmission fails %d \n", + err); + return; + } +} + +static int avb_avdecc_listen(struct avdecc *avdecc) +{ + int err = 0; + mm_segment_t oldfs; + struct kvec vec; + + memset(avdecc->sd.rx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + avdecc->sd.rx_iov.iov_base = avdecc->sd.rx_buf; + avdecc->sd.rx_iov.iov_len = AVB_MAX_ETH_FRAME_SIZE; + iov_iter_init(&avdecc->sd.rx_msg_hdr.msg_iter, READ, &avdecc->sd.rx_iov, + 1, AVB_MAX_ETH_FRAME_SIZE); + + vec.iov_base = avdecc->sd.rx_buf; + vec.iov_len = AVB_MAX_ETH_FRAME_SIZE; + + oldfs = get_fs(); + set_fs(KERNEL_DS); + err = kernel_recvmsg(avdecc->sd.sock, &avdecc->sd.rx_msg_hdr, &vec, 1, + AVB_MAX_ETH_FRAME_SIZE, MSG_DONTWAIT); + set_fs(oldfs); + + if (err <= 0) + if (err != -11) + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_adecc_listen Socket reception res %d \n", + err); + + return err; +} + +static void avb_avdecc_aecp_respond_to_AEM_cmd(struct avdecc *avdecc, + struct msrp *msrp) +{ + int i = 0; + int err = 0; + u16 tx_size = 0; + int descp_size = 0; + int max_cfg_idx = 0; + int max_desc_idx = 0; + struct aem_cmd *cmd = + (struct aem_cmd *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + struct read_descp_cmd *rd_cmd = (struct read_descp_cmd *)cmd; + struct get_stream_info_cmd *gt_str_info_cmd = + (struct get_stream_info_cmd *)cmd; + struct read_descp_res *rd_res = + (struct read_descp_res *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_acdecc_init_and_fill_eth_hdr(avdecc, 0); + + memcpy(&rd_res->hdr.ctrl_entity_id[0], &rd_cmd->hdr.ctrl_entity_id[0], + 8); + rd_res->hdr.seq_id = rd_cmd->hdr.seq_id; + rd_res->hdr.cmd_type = cmd->cmd_type; + + switch (htons(cmd->cmd_type)) { + case AVB_AEM_CMD_READ_DESCP: { + rd_res->cfg_idx = rd_cmd->cfg_idx; + rd_res->res = 0; + + switch (htons(rd_cmd->desc_type)) { + case AVB_AEM_DESCP_ENTITY: { + struct entity_descp *ent_descp = + (struct entity_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Entity Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct entity_descp)), + 1); + ent_descp->desc_type = htons(AVB_AEM_DESCP_ENTITY); + ent_descp->desc_idx = 0; + ent_descp->entity_id[0] = avdecc->sd.srcmac[0]; + ent_descp->entity_id[1] = avdecc->sd.srcmac[1]; + ent_descp->entity_id[2] = avdecc->sd.srcmac[2]; + ent_descp->entity_id[3] = 0xff; + ent_descp->entity_id[4] = 0xfe; + ent_descp->entity_id[5] = avdecc->sd.srcmac[3]; + ent_descp->entity_id[6] = avdecc->sd.srcmac[4]; + ent_descp->entity_id[7] = avdecc->sd.srcmac[5]; + ent_descp->entity_model_id[0] = avdecc->sd.srcmac[0]; + ent_descp->entity_model_id[1] = avdecc->sd.srcmac[1]; + ent_descp->entity_model_id[2] = avdecc->sd.srcmac[2]; + ent_descp->entity_model_id[3] = avdecc->sd.srcmac[3]; + ent_descp->entity_model_id[4] = avdecc->sd.srcmac[4]; + ent_descp->entity_model_id[5] = avdecc->sd.srcmac[5]; + ent_descp->entity_model_id[6] = 0x00; + ent_descp->entity_model_id[7] = 0x01; + ent_descp->entity_caps = htonl(0x00008508); + ent_descp->talker_stream_sources = htons(1); + ent_descp->talker_caps = htons(0x4001); + ent_descp->listener_stream_sinks = htons(1); + ent_descp->listener_caps = htons(0x4001); + ent_descp->control_caps = 0; + ent_descp->avai_idx = htonl(avdecc->adp_avai_idx); + strcpy(&ent_descp->entity_name[0], "ALSA_AVB_Driver"); + strcpy(&ent_descp->firmware_ver[0], "0.1"); + strcpy(&ent_descp->group_name[0], "0"); + strcpy(&ent_descp->serial_number[0], "0"); + ent_descp->vendor_name_string = htons(0x0000); + ent_descp->model_name_string = htons(0x0001); + ent_descp->cfg_count = htons(1); + ent_descp->curr_cfg = htons(0); + descp_size = sizeof(struct entity_descp); + break; + } + case AVB_AEM_DESCP_CONFIGURATION: { + struct config_descp *cfg_descp = + (struct config_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Configuration Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct config_descp)), + 1); + cfg_descp->desc_type = + htons(AVB_AEM_DESCP_CONFIGURATION); + cfg_descp->desc_idx = 0; + cfg_descp->localized_descp = htons(0xffff); + cfg_descp->descp_count = htons(AVB_AEM_MAX_DESCP_COUNT); + cfg_descp->descpOff = htons(74); + + cfg_descp->descps[0].desc_type = + htons(AVB_AEM_DESCP_AUDIO_UNIT); + cfg_descp->descps[0].desc_count = htons(1); + cfg_descp->descps[1].desc_type = + htons(AVB_AEM_DESCP_STREAM_IP); + cfg_descp->descps[1].desc_count = htons(1); + cfg_descp->descps[2].desc_type = + htons(AVB_AEM_DESCP_STREAM_OP); + cfg_descp->descps[2].desc_count = htons(1); + cfg_descp->descps[3].desc_type = + htons(AVB_AEM_DESCP_AVBINTERFACE); + cfg_descp->descps[3].desc_count = htons(1); + cfg_descp->descps[4].desc_type = + htons(AVB_AEM_DESCP_CLOCKSOURCE); + cfg_descp->descps[4].desc_count = htons(2); + cfg_descp->descps[5].desc_type = + htons(AVB_AEM_DESCP_LOCALE); + cfg_descp->descps[5].desc_count = htons(1); + cfg_descp->descps[6].desc_type = + htons(AVB_AEM_DESCP_CLOCK_DOMAIN); + cfg_descp->descps[6].desc_count = htons(1); + descp_size = sizeof(struct config_descp); + break; + } + case AVB_AEM_DESCP_AUDIO_UNIT: { + struct audio_unit_descp *au_descp = + (struct audio_unit_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Audiounit Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct audio_unit_descp)), + 1); + au_descp->desc_type = htons(AVB_AEM_DESCP_AUDIO_UNIT); + au_descp->desc_idx = 0; + au_descp->localized_descp = htons(0x0001); + + au_descp->num_stream_ip = htons(1); + au_descp->num_stream_op = htons(1); + au_descp->num_ext_ip = htons(8); + au_descp->num_ext_op = htons(8); + au_descp->current_sampling_rate = htonl(48000); + au_descp->sampling_rates_offset = htons(144); + au_descp->sampling_rates_count = htons(6); + au_descp->sampling_rates[0] = htonl(44100); + au_descp->sampling_rates[1] = htonl(48000); + au_descp->sampling_rates[2] = htonl(88200); + au_descp->sampling_rates[3] = htonl(96000); + au_descp->sampling_rates[4] = htonl(176400); + au_descp->sampling_rates[5] = htonl(192000); + descp_size = sizeof(struct audio_unit_descp); + break; + } + case AVB_AEM_DESCP_STREAM_PORT_IP: + case AVB_AEM_DESCP_STREAM_PORT_OP: { + struct stream_port_descp *st_port_descp = + (struct stream_port_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + if (htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_STREAM_PORT_IP) + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Stream Input Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + else + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Stream Output Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct stream_port_descp)), + 1); + st_port_descp->desc_type = rd_cmd->desc_type; + st_port_descp->desc_idx = rd_cmd->desc_idx; + + st_port_descp->num_clusters = htons(8); + st_port_descp->num_maps = htons(1); + st_port_descp->port_flags = + htons(((htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_STREAM_PORT_IP) ? + (0) : + (1))); + st_port_descp->base_cluster = + htons(((htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_STREAM_PORT_IP) ? + (0) : + (8))); + st_port_descp->base_map = + htons(((htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_STREAM_PORT_IP) ? + (0) : + (1))); + descp_size = sizeof(struct stream_port_descp); + break; + } + case AVB_AEM_DESCP_EXT_PORT_IP: + case AVB_AEM_DESCP_EXT_PORT_OP: { + struct ext_port_descp *ext_port_descp = + (struct ext_port_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + if (htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_EXT_PORT_IP) + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for External Input Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + else + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for External Output Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct ext_port_descp)), + 1); + ext_port_descp->desc_type = rd_cmd->desc_type; + ext_port_descp->desc_idx = rd_cmd->desc_idx; + + ext_port_descp->signal_type = + htons(((htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_EXT_PORT_IP) ? + (AVB_AEM_DESCP_INVALID) : + (AVB_AEM_DESCP_AUDIO_CLUSTER))); + ext_port_descp->signal_idx = + htons(((htons(rd_cmd->desc_type) == + AVB_AEM_DESCP_EXT_PORT_IP) ? + (0) : + (htons(rd_cmd->desc_idx)))); + descp_size = sizeof(struct ext_port_descp); + max_desc_idx = 7; + break; + } + case AVB_AEM_DESCP_JACK_IP: + case AVB_AEM_DESCP_JACK_OP: { + struct jack_descp *jack_descp = + (struct jack_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + if (htons(rd_cmd->desc_type) == AVB_AEM_DESCP_JACK_IP) + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Jack Input Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + else + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Jack Output Port (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct jack_descp)), + 1); + jack_descp->desc_type = rd_cmd->desc_type; + jack_descp->desc_idx = rd_cmd->desc_idx; + if (htons(rd_cmd->desc_type) == AVB_AEM_DESCP_JACK_IP) + strcpy(&jack_descp->obj_name[0], "ip-"); + else + strcpy(&jack_descp->obj_name[0], "op-"); + + jack_descp->obj_name[3] = 48 + htons(rd_cmd->desc_idx); + jack_descp->localized_descp = htons(0x0007); + + jack_descp->jack_type = htons(8); /* Balanced analog */ + descp_size = sizeof(struct jack_descp); + max_desc_idx = 7; + break; + } + case AVB_AEM_DESCP_AUDIO_CLUSTER: { + struct audio_cluster_descp *au_cl_descp = + (struct audio_cluster_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Audio Cluster (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct audio_cluster_descp)), + 1); + au_cl_descp->desc_type = rd_cmd->desc_type; + au_cl_descp->desc_idx = rd_cmd->desc_idx; + + au_cl_descp->localized_descp = htons(0xffff); + + au_cl_descp->signal_type = + htons(((htons(rd_cmd->desc_idx) < 8) ? + (AVB_AEM_DESCP_INVALID) : + (AVB_AEM_DESCP_EXT_PORT_IP))); + au_cl_descp->signal_idx = + htons(((htons(rd_cmd->desc_idx) < 8) ? + (0) : + (htons(rd_cmd->desc_idx) - 8))); + au_cl_descp->num_channels = htons(1); + au_cl_descp->format = 0x40; + descp_size = sizeof(struct audio_cluster_descp); + max_desc_idx = 15; + break; + } + case AVB_AEM_DESCP_AUDIO_MAP: { + struct audio_map_descp *au_map_descp = + (struct audio_map_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Audio Map (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct audio_map_descp)), + 1); + au_map_descp->desc_type = rd_cmd->desc_type; + au_map_descp->desc_idx = rd_cmd->desc_idx; + + au_map_descp->mapping_offset = htons(8); + au_map_descp->num_mappings = htons(8); + + for (i = 0; i < 8; i++) { + au_map_descp->map[i].stream_idx = htons(0); + au_map_descp->map[i].stream_channel = htons(i); + au_map_descp->map[i].cluster_offset = htons(i); + au_map_descp->map[i].cluster_channel = htons(0); + } + descp_size = sizeof(struct audio_map_descp); + max_desc_idx = 1; + break; + } + case AVB_AEM_DESCP_CLOCKSOURCE: { + struct clock_source_descp *clk_src_descp = + (struct clock_source_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Clock Source (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct clock_source_descp)), + 1); + clk_src_descp->desc_type = rd_cmd->desc_type; + clk_src_descp->desc_idx = rd_cmd->desc_idx; + + clk_src_descp->localized_descp = + htons(0x0004 + htons(rd_cmd->desc_idx)); + + clk_src_descp->clock_source_id[0] = + avdecc->sd.srcmac[0]; + clk_src_descp->clock_source_id[1] = + avdecc->sd.srcmac[1]; + clk_src_descp->clock_source_id[2] = + avdecc->sd.srcmac[2]; + clk_src_descp->clock_source_id[3] = + avdecc->sd.srcmac[3]; + clk_src_descp->clock_source_id[4] = + avdecc->sd.srcmac[4]; + clk_src_descp->clock_source_id[5] = + avdecc->sd.srcmac[5]; + clk_src_descp->clock_source_id[6] = 0; + clk_src_descp->clock_source_id[7] = + (u8)htons(rd_cmd->desc_idx); + + if (htons(rd_cmd->desc_idx) == 0) { + clk_src_descp->clock_source_flags = htons(0x00); + clk_src_descp->clock_source_type = htons(0x00); + clk_src_descp->clock_source_loc_type = + htons(AVB_AEM_DESCP_AUDIO_UNIT); + } else if (htons(rd_cmd->desc_idx) == 1) { + clk_src_descp->clock_source_flags = htons(0x00); + clk_src_descp->clock_source_type = htons(0x01); + clk_src_descp->clock_source_loc_type = + htons(AVB_AEM_DESCP_JACK_IP); + } else { + clk_src_descp->clock_source_flags = htons(0x02); + clk_src_descp->clock_source_type = htons(0x02); + clk_src_descp->clock_source_loc_type = + htons(AVB_AEM_DESCP_STREAM_IP); + } + + descp_size = sizeof(struct clock_source_descp); + max_desc_idx = 2; + break; + } + case AVB_AEM_DESCP_CLOCK_DOMAIN: { + struct clock_domain_descp *clk_do_descp = + (struct clock_domain_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_aecp_readResponse for Clock Domain (%d) Descriptor", + htons(rd_cmd->desc_idx)); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct clock_domain_descp)), + 1); + clk_do_descp->desc_type = rd_cmd->desc_type; + clk_do_descp->desc_idx = rd_cmd->desc_idx; + + clk_do_descp->localized_descp = htons(0x0000); + + clk_do_descp->curr_clock_source = htons(0x00); + clk_do_descp->clock_sources_offset = htons(0x4c); + clk_do_descp->clock_sources_count = htons(3); + clk_do_descp->clock_sources[0] = htons(0); + clk_do_descp->clock_sources[1] = htons(1); + clk_do_descp->clock_sources[2] = htons(2); + + descp_size = sizeof(struct clock_domain_descp); + break; + } + case AVB_AEM_DESCP_AVBINTERFACE: { + struct avb_if_descp *avb_if_descp = + (struct avb_if_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for AVB Interface Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct avb_if_descp)), + 1); + avb_if_descp->desc_type = + htons(AVB_AEM_DESCP_AVBINTERFACE); + avb_if_descp->desc_idx = 0; + avb_if_descp->localized_descp = htons(0xffff); + avb_if_descp->mac_addr[0] = avdecc->sd.srcmac[0]; + avb_if_descp->mac_addr[1] = avdecc->sd.srcmac[1]; + avb_if_descp->mac_addr[2] = avdecc->sd.srcmac[2]; + avb_if_descp->mac_addr[3] = avdecc->sd.srcmac[3]; + avb_if_descp->mac_addr[4] = avdecc->sd.srcmac[4]; + avb_if_descp->mac_addr[5] = avdecc->sd.srcmac[5]; + avb_if_descp->if_flags = htons(0x0007); + avb_if_descp->clock_iden[0] = avdecc->sd.srcmac[0]; + avb_if_descp->clock_iden[1] = avdecc->sd.srcmac[1]; + avb_if_descp->clock_iden[2] = avdecc->sd.srcmac[2]; + avb_if_descp->clock_iden[3] = 0xff; + avb_if_descp->clock_iden[4] = 0xfe; + avb_if_descp->clock_iden[5] = avdecc->sd.srcmac[3]; + avb_if_descp->clock_iden[6] = avdecc->sd.srcmac[4]; + avb_if_descp->clock_iden[7] = avdecc->sd.srcmac[5]; + avb_if_descp->prio1 = 250; + avb_if_descp->clock_class = 248; + avb_if_descp->off_scaled_log_var = htons(0x4100); + avb_if_descp->clock_accu = 254; + avb_if_descp->prio2 = 250; + avb_if_descp->domain_no = 0; + avb_if_descp->log_sync_int = 15; + avb_if_descp->log_anno_int = 15; + avb_if_descp->log_pdelay_int = 13; + avb_if_descp->port_no = htons(0x0001); + descp_size = sizeof(struct avb_if_descp); + break; + } + case AVB_AEM_DESCP_LOCALE: { + struct locale_descp *locale_descp = + (struct locale_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Locale Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct locale_descp)), + 1); + + locale_descp->desc_type = htons(AVB_AEM_DESCP_LOCALE); + locale_descp->desc_idx = 0; + strcpy(&locale_descp->locale_id[0], "en-US"); + locale_descp->num_strings = htons(1); + locale_descp->base_strings_idx = htons(0); + descp_size = sizeof(struct locale_descp); + break; + } + case AVB_AEM_DESCP_STRINGS: { + struct strings_descp *strings_descp = + (struct strings_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Strings Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct strings_descp)), + 1); + + strings_descp->desc_type = htons(AVB_AEM_DESCP_STRINGS); + strings_descp->desc_idx = 0; + strcpy(&strings_descp->strings[0][0], "FH-Kiel"); + strcpy(&strings_descp->strings[1][0], "BBB"); + strcpy(&strings_descp->strings[2][0], "Stream IP"); + strcpy(&strings_descp->strings[3][0], "Stream OP"); + strcpy(&strings_descp->strings[4][0], "Domain Clock"); + strcpy(&strings_descp->strings[5][0], "External Clock"); + strcpy(&strings_descp->strings[6][0], "Stream Clock"); + descp_size = sizeof(struct strings_descp); + break; + } + case AVB_AEM_DESCP_STREAM_IP: { + struct stream_descp *strm_op_descp = + (struct stream_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Stream Out Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct stream_descp)), + 1); + strm_op_descp->desc_type = + htons(AVB_AEM_DESCP_STREAM_IP); + strm_op_descp->desc_idx = 0; + strm_op_descp->localized_descp = htons(0x0002); + strm_op_descp->clock_domain_idx = 0; + strm_op_descp->stream_flags = htons(0x0003); + strm_op_descp->avb_if_idx = 0; + strm_op_descp->buf_size = htonl(583333); + strm_op_descp->fmts_count = + htons(AVB_AEM_MAX_SUPP_FORMATS); + strm_op_descp->fmts_off = htons(132); + +#if 0 + strm_op_descp->supp_fmts[0].fmt.avtp.sub_type = AVB_AEM_STREAM_FORMAT_AVTP; + strm_op_descp->supp_fmts[0].fmt.avtp.b1.nsr = 5; + strm_op_descp->supp_fmts[0].fmt.avtp.format = 4; + strm_op_descp->supp_fmts[0].fmt.avtp.bit_depth = 16; + strm_op_descp->supp_fmts[0].fmt.avtp.cpf = 2; + strm_op_descp->supp_fmts[0].fmt.avtp.b5.cpf = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.b5.spf = strm_op_descp->supp_fmts[0].fmt.avtp.b5.spf | 0x01; + strm_op_descp->supp_fmts[0].fmt.avtp.b6.spf = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.b6.res2 = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.res2 = 0; +#else + strm_op_descp->supp_fmts[0].fmt.iec.sub_type = 0; + strm_op_descp->supp_fmts[0].fmt.iec.b1.sf = 0x80; + strm_op_descp->supp_fmts[0].fmt.iec.b1.fmt = + strm_op_descp->supp_fmts[0].fmt.iec.b1.fmt | + 0x20; + strm_op_descp->supp_fmts[0].fmt.iec.b2.fdf_sfc = 0x01; + strm_op_descp->supp_fmts[0].fmt.iec.dbs = 0x08; + strm_op_descp->supp_fmts[0].fmt.iec.b4.b = 0x60; + strm_op_descp->supp_fmts[0].fmt.iec.label_mbla_cnt = + 0x08; +#endif + memcpy(&strm_op_descp->supp_fmts[1], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[2], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[3], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[4], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[5], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + + strm_op_descp->supp_fmts[1].fmt.iec.b2.fdf_sfc = 0x02; + strm_op_descp->supp_fmts[2].fmt.iec.b2.fdf_sfc = 0x03; + strm_op_descp->supp_fmts[3].fmt.iec.b2.fdf_sfc = 0x04; + strm_op_descp->supp_fmts[4].fmt.iec.b2.fdf_sfc = 0x05; + strm_op_descp->supp_fmts[5].fmt.iec.b2.fdf_sfc = 0x06; + + memcpy(&strm_op_descp->curr_fmt, + &strm_op_descp->supp_fmts[1], + sizeof(struct stream_format)); + + descp_size = sizeof(struct stream_descp); + break; + } + case AVB_AEM_DESCP_STREAM_OP: { + struct stream_descp *strm_op_descp = + (struct stream_descp *)&avdecc->sd.tx_buf + [sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Stream Out Descriptor"); + + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct read_descp_res) + + sizeof(struct stream_descp)), + 1); + strm_op_descp->desc_type = + htons(AVB_AEM_DESCP_STREAM_OP); + strm_op_descp->desc_idx = 0; + strm_op_descp->localized_descp = htons(0x0003); + strm_op_descp->clock_domain_idx = 0; + strm_op_descp->stream_flags = htons(0x0002); + strm_op_descp->avb_if_idx = 0; + strm_op_descp->buf_size = htonl(583333); + strm_op_descp->fmts_count = + htons(AVB_AEM_MAX_SUPP_FORMATS); + strm_op_descp->fmts_off = htons(132); + +#if 0 + strm_op_descp->supp_fmts[0].fmt.avtp.sub_type = AVB_AEM_STREAM_FORMAT_AVTP; + strm_op_descp->supp_fmts[0].fmt.avtp.b1.nsr = 5; + strm_op_descp->supp_fmts[0].fmt.avtp.format = 4; + strm_op_descp->supp_fmts[0].fmt.avtp.bit_depth = 16; + strm_op_descp->supp_fmts[0].fmt.avtp.cpf = 2; + strm_op_descp->supp_fmts[0].fmt.avtp.b5.cpf = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.b5.spf = strm_op_descp->supp_fmts[0].fmt.avtp.b5.spf | 0x01; + strm_op_descp->supp_fmts[0].fmt.avtp.b6.spf = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.b6.res2 = 0; + strm_op_descp->supp_fmts[0].fmt.avtp.res2 = 0; +#else + strm_op_descp->supp_fmts[0].fmt.iec.sub_type = 0; + strm_op_descp->supp_fmts[0].fmt.iec.b1.sf = 0x80; + strm_op_descp->supp_fmts[0].fmt.iec.b1.fmt = + strm_op_descp->supp_fmts[0].fmt.iec.b1.fmt | + 0x20; + strm_op_descp->supp_fmts[0].fmt.iec.b2.fdf_sfc = 0x01; + strm_op_descp->supp_fmts[0].fmt.iec.dbs = 0x08; + strm_op_descp->supp_fmts[0].fmt.iec.b4.b = 0x60; + strm_op_descp->supp_fmts[0].fmt.iec.label_mbla_cnt = + 0x08; +#endif + memcpy(&strm_op_descp->supp_fmts[1], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[2], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[3], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[4], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + memcpy(&strm_op_descp->supp_fmts[5], + &strm_op_descp->supp_fmts[0], + sizeof(struct stream_format)); + + strm_op_descp->supp_fmts[1].fmt.iec.b2.fdf_sfc = 0x02; + strm_op_descp->supp_fmts[2].fmt.iec.b2.fdf_sfc = 0x03; + strm_op_descp->supp_fmts[3].fmt.iec.b2.fdf_sfc = 0x04; + strm_op_descp->supp_fmts[4].fmt.iec.b2.fdf_sfc = 0x05; + strm_op_descp->supp_fmts[5].fmt.iec.b2.fdf_sfc = 0x06; + + memcpy(&strm_op_descp->curr_fmt, + &strm_op_descp->supp_fmts[1], + sizeof(struct stream_format)); + + descp_size = sizeof(struct stream_descp); + break; + } + default: { + struct read_descp_cmd *rd_res_cmd = + (struct read_descp_cmd *)rd_res; + rd_res_cmd->desc_type = rd_cmd->desc_type; + rd_res_cmd->desc_idx = rd_cmd->desc_idx; + descp_size = 0; + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_NOT_IMPLEMENTED, + (sizeof(struct read_descp_cmd)), 1); + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_aecp_respond_to_AEM_cmd unknown desc_type: %d", + htons(rd_cmd->desc_type)); + break; + } + } + + if (descp_size > 0) { + if ((htons(rd_cmd->cfg_idx) > max_cfg_idx) || + (htons(rd_cmd->desc_idx) > max_desc_idx)) { + struct read_descp_cmd *rd_res_cmd = + (struct read_descp_cmd *)rd_res; + rd_res_cmd->desc_type = rd_cmd->desc_type; + rd_res_cmd->desc_idx = rd_cmd->desc_idx; + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_NO_SUCH_DESCRIPTOR, + (sizeof(struct read_descp_cmd)), 1); + } else { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res) + + descp_size; + } + } + + break; + } + case AVB_AEM_CMD_ENTITY_ACQUIRE: { + struct acquire_ent_cmd *acq_ent_cmd = + (struct acquire_ent_cmd *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_aecp_readResponse for Entiry Acquire"); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acquire_ent_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct acquire_ent_cmd)), + 1); + + memcpy(&acq_ent_cmd->owner_id[0], + &acq_ent_cmd->hdr.ctrl_entity_id[0], 8); + acq_ent_cmd->flags = 0; + acq_ent_cmd->desc_type = htons(AVB_AEM_DESCP_ENTITY); + acq_ent_cmd->desc_idx = 0; + + msrp->started = true; + avb_msrp_talkerdeclarations(msrp, true, msrp->tx_state); + break; + } + case AVB_AEM_CMD_ENTITY_AVAILABLE: { + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_aecp_readResponse for Entity Available"); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct aem_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct aem_cmd)), 1); + break; + } + case AVB_AEM_CMD_SET_STREAM_FORMAT: { + struct set_stream_format_cmd *reqStFmt = + (struct set_stream_format_cmd *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + struct set_stream_format_cmd *stFmt = + (struct set_stream_format_cmd *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Set Stream Format"); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct set_stream_format_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, AVB_AEM_RES_SUCCESS, + (sizeof(struct set_stream_format_cmd)), 1); + memcpy(stFmt, reqStFmt, sizeof(struct set_stream_format_cmd)); + break; + } + case AVB_AEM_CMD_GET_STREAM_INFO: { + struct aem_stream_info *get_stream_info_req = + (struct aem_stream_info *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + struct aem_stream_info *aem_stream_info = + (struct aem_stream_info *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Get Stream Info command"); + + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + sizeof(struct aem_stream_info), + 1); + aem_stream_info->desc_type = get_stream_info_req->desc_type; + aem_stream_info->desc_idx = 0; + aem_stream_info->flags = htonl(0x0); + aem_stream_info->curr_fmt.fmt.avtp.sub_type = + AVB_AEM_STREAM_FORMAT_AVTP; + aem_stream_info->curr_fmt.fmt.avtp.b1.nsr = 5; + aem_stream_info->curr_fmt.fmt.avtp.format = 4; + aem_stream_info->curr_fmt.fmt.avtp.bit_depth = 16; + aem_stream_info->curr_fmt.fmt.avtp.cpf = 2; + aem_stream_info->curr_fmt.fmt.avtp.b5.cpf = 0; + aem_stream_info->curr_fmt.fmt.avtp.b5.spf = + aem_stream_info->curr_fmt.fmt.avtp.b5.spf | 0x01; + aem_stream_info->curr_fmt.fmt.avtp.b6.spf = 0; + aem_stream_info->curr_fmt.fmt.avtp.b6.res2 = 0; + aem_stream_info->curr_fmt.fmt.avtp.res2 = 0; + aem_stream_info->stream_id[0] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[0]) : + (0)); + aem_stream_info->stream_id[1] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[1]) : + (0)); + aem_stream_info->stream_id[2] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[2]) : + (0)); + aem_stream_info->stream_id[3] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[3]) : + (0)); + aem_stream_info->stream_id[4] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[4]) : + (0)); + aem_stream_info->stream_id[5] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (avdecc->sd.srcmac[5]) : + (0)); + aem_stream_info->stream_id[6] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (0) : + (0)); + aem_stream_info->stream_id[7] = + ((htons(get_stream_info_req->desc_type) == + AVB_AEM_DESCP_STREAM_OP) ? + (1) : + (0)); + aem_stream_info->msrp_accu_lat = htonl(0); + aem_stream_info->stream_dest_MAC[0] = 0x91; + aem_stream_info->stream_dest_MAC[1] = 0xe0; + aem_stream_info->stream_dest_MAC[2] = 0xf0; + aem_stream_info->stream_dest_MAC[3] = 0x00; + aem_stream_info->stream_dest_MAC[4] = 0x3f; + aem_stream_info->stream_dest_MAC[5] = 0xf1; + aem_stream_info->msrp_failure_code = 0; + aem_stream_info->stream_vlan_id = htons(2); + + if (gt_str_info_cmd->desc_idx > 0) { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct get_stream_info_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_NO_SUCH_DESCRIPTOR, + (sizeof(struct get_stream_info_cmd)), 1); + } else { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct aem_stream_info); + } + break; + } + case AVB_AEM_CMD_GET_COUNTERS: { + struct counters_descp *counters_req = + (struct counters_descp *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + struct counters_descp *counters_descp = + (struct counters_descp *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Get Counters command"); + + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + sizeof(struct aem_stream_info), + 1); + + counters_descp->desc_type = counters_req->desc_type; + counters_descp->desc_idx = 0; + + if (counters_req->desc_type == AVB_AEM_DESCP_STREAM_IP) { + counters_descp->counters_valid = htonl(0x00001800); + + counters_descp->counters[11] = 0; + counters_descp->counters[12] = 0; + } else { + counters_descp->counters_valid = htonl(0x0); + } + + if (gt_str_info_cmd->desc_idx > 0) { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct get_counters_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_NO_SUCH_DESCRIPTOR, + (sizeof(struct get_counters_cmd)), 1); + } else { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct counters_descp); + } + break; + } + case AVB_AEM_CMD_REGISTER_UNSOLICITED_NOTIFICATION: { + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_aecp_readResponse for Register Unsolicited Responses"); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct aem_cmd); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_SUCCESS, + (sizeof(struct aem_cmd)), 1); + break; + } + default: { + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct read_descp_res); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_AECP, + AVB_AECP_MSGTYPE_AEM_RESPONSE, + AVB_AEM_RES_NOT_IMPLEMENTED, + (sizeof(struct read_descp_res)), + 1); + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_aecp_respond_to_AEM_cmd unknown cmd_type: %d", + htons(cmd->cmd_type)); + break; + } + } + + if (tx_size > 0) { + avdecc->sd.tx_iov.iov_base = avdecc->sd.tx_buf; + avdecc->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avdecc->sd.tx_msg_hdr.msg_iter, WRITE, + &avdecc->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avdecc->sd.sock, + &avdecc->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_avdecc_aecp_respond_to_AEM_cmd Socket transmission fails %d \n", + err); + return; + } + } +} + +static void avb_avdecc_aecp_respond_to_cmd(struct avdecc *avdecc, + struct msrp *msrp) +{ + struct avt_pdu_control_hdr *hdr = + (struct avt_pdu_control_hdr *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr)]; + + switch (hdr->h.f.b1.msg_type) { + case AVB_AECP_MSGTYPE_AEM_COMMAND: + avb_avdecc_aecp_respond_to_AEM_cmd(avdecc, msrp); + break; + + default: + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_aecp_respond_to_cmd unknown sub_type: %d", + hdr->h.f.b1.msg_type); + break; + } +} + +static void avb_avdecc_acmp_respond_to_cmd(struct avdecc *avdecc) +{ + int err = 0; + int dest = 0; + u16 tx_size = 0; + struct avt_pdu_control_hdr *hdr = + (struct avt_pdu_control_hdr *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr)]; + struct acm_pdu *rx_acmp_du = + (struct acm_pdu *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + struct acm_pdu *tx_acmp_du = + (struct acm_pdu *)&avdecc->sd + .tx_buf[sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr)]; + + if ((rx_acmp_du->talker_entity_id[0] == avdecc->sd.srcmac[0]) && + (rx_acmp_du->talker_entity_id[1] == avdecc->sd.srcmac[1]) && + (rx_acmp_du->talker_entity_id[2] == avdecc->sd.srcmac[2]) && + (rx_acmp_du->talker_entity_id[3] == 0xff) && + (rx_acmp_du->talker_entity_id[4] == 0xfe) && + (rx_acmp_du->talker_entity_id[5] == avdecc->sd.srcmac[3]) && + (rx_acmp_du->talker_entity_id[6] == avdecc->sd.srcmac[4]) && + (rx_acmp_du->talker_entity_id[7] == avdecc->sd.srcmac[5])) { + dest++; + } else if ((rx_acmp_du->listener_entity_id[0] == + avdecc->sd.srcmac[0]) && + (rx_acmp_du->listener_entity_id[1] == + avdecc->sd.srcmac[1]) && + (rx_acmp_du->listener_entity_id[2] == + avdecc->sd.srcmac[2]) && + (rx_acmp_du->listener_entity_id[3] == 0xff) && + (rx_acmp_du->listener_entity_id[4] == 0xfe) && + (rx_acmp_du->listener_entity_id[5] == + avdecc->sd.srcmac[3]) && + (rx_acmp_du->listener_entity_id[6] == + avdecc->sd.srcmac[4]) && + (rx_acmp_du->listener_entity_id[7] == + avdecc->sd.srcmac[5])) { + dest--; + } else { + dest = 0; + } + + if (((hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_GET_TX_STATE_CMD) || + (hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_CONNECT_TX_CMD) || + (hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_DISCONNECT_TX_CMD) || + (hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_GET_TX_CONN_CMD)) && + (dest <= 0)) { + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_aecp_respond_to_cmd ACMP TX cmd: %d not for us", + hdr->h.f.b1.msg_type); + return; + } + + if (((hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_GET_RX_STATE_CMD) || + (hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_CONNECT_RX_CMD) || + (hdr->h.f.b1.msg_type == AVB_ACMP_MSGTYPE_DISCONNECT_RX_CMD)) && + (dest >= 0)) { + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_aecp_respond_to_cmd ACMP RX cmd: %d not for us", + hdr->h.f.b1.msg_type); + return; + } + + switch (hdr->h.f.b1.msg_type) { + case AVB_ACMP_MSGTYPE_GET_TX_STATE_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Get TX state command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_GET_TX_STATE_RESP, + avdecc->acmp_tx_state, sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->stream_dest_MAC[0] = 0x91; + tx_acmp_du->stream_dest_MAC[1] = 0xe0; + tx_acmp_du->stream_dest_MAC[2] = 0xf0; + tx_acmp_du->stream_dest_MAC[3] = 0x00; + tx_acmp_du->stream_dest_MAC[4] = 0x33; + tx_acmp_du->stream_dest_MAC[5] = 0x4b; + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + case AVB_ACMP_MSGTYPE_GET_RX_STATE_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Get RX state command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_GET_RX_STATE_RESP, + avdecc->acmp_rx_state, sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->stream_dest_MAC[0] = 0x91; + tx_acmp_du->stream_dest_MAC[1] = 0xe0; + tx_acmp_du->stream_dest_MAC[2] = 0xf0; + tx_acmp_du->stream_dest_MAC[3] = 0x00; + tx_acmp_du->stream_dest_MAC[4] = 0x33; + tx_acmp_du->stream_dest_MAC[5] = 0x4c; + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + case AVB_ACMP_MSGTYPE_CONNECT_TX_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Connect TX command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_CONNECT_TX_RESP, + AVB_ACMP_STATUS_SUCCESS, + sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->stream_dest_MAC[0] = 0x91; + tx_acmp_du->stream_dest_MAC[1] = 0xe0; + tx_acmp_du->stream_dest_MAC[2] = 0xf0; + tx_acmp_du->stream_dest_MAC[3] = 0x00; + tx_acmp_du->stream_dest_MAC[4] = 0x33; + tx_acmp_du->stream_dest_MAC[5] = 0x4b; + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + case AVB_ACMP_MSGTYPE_DISCONNECT_TX_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Disconnect TX command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_DISCONNECT_TX_RESP, + AVB_ACMP_STATUS_SUCCESS, sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + case AVB_ACMP_MSGTYPE_CONNECT_RX_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Connect RX command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr(avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_CONNECT_RX_RESP, + AVB_ACMP_STATUS_SUCCESS, + sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->stream_dest_MAC[0] = 0x91; + tx_acmp_du->stream_dest_MAC[1] = 0xe0; + tx_acmp_du->stream_dest_MAC[2] = 0xf0; + tx_acmp_du->stream_dest_MAC[3] = 0x00; + tx_acmp_du->stream_dest_MAC[4] = 0x33; + tx_acmp_du->stream_dest_MAC[5] = 0x4c; + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + case AVB_ACMP_MSGTYPE_DISCONNECT_RX_CMD: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Disconnect RX command"); + avb_acdecc_init_and_fill_eth_hdr(avdecc, 1); + avb_acdecc_fill_AVTP_ctrl_hdr( + avdecc, AVB_AVTP_SUBTYPE_ACMP, + AVB_ACMP_MSGTYPE_DISCONNECT_RX_RESP, + AVB_ACMP_STATUS_SUCCESS, sizeof(struct acm_pdu), 0); + memcpy(tx_acmp_du, rx_acmp_du, sizeof(struct acm_pdu)); + tx_acmp_du->flags = htons(0x0000); + tx_size = sizeof(struct ethhdr) + + sizeof(struct avt_pdu_control_hdr) + + sizeof(struct acm_pdu); + break; + + default: + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_acmp_respond_to_cmd unknown sub_type: %d", + hdr->h.f.b1.msg_type); + break; + } + + if (tx_size > 0) { + avdecc->sd.tx_iov.iov_base = avdecc->sd.tx_buf; + avdecc->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&avdecc->sd.tx_msg_hdr.msg_iter, WRITE, + &avdecc->sd.tx_iov, 1, tx_size); + + if ((err = sock_sendmsg(avdecc->sd.sock, + &avdecc->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_avdecc_acmp_respondToAEMCmd Socket transmission fails %d \n", + err); + return; + } + } +} + +static void avb_avdecc_adp_respond_to_cmd(struct avdecc *avdecc) +{ + struct avt_pdu_control_hdr *hdr = + (struct avt_pdu_control_hdr *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr)]; + + switch (hdr->h.f.b1.msg_type) { + case AVB_ADP_MSGTYPE_ENTITY_AVAILABLE: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Entity Available"); + break; + + case AVB_ADP_MSGTYPE_ENTITY_DEPARTING: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Entity Departing"); + break; + + case AVB_ADP_MSGTYPE_ENTITY_DISCOVER: + avb_log(AVB_KERN_INFO, KERN_INFO + "avb_avdecc_adp_respond_to_cmd Entity Discover"); + break; + + default: + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_adp_respond_to_cmd unknown sub_type: %d", + hdr->h.f.b1.msg_type); + break; + } +} + +void avb_avdecc_listen_and_respond(struct avdecc *avdecc, struct msrp *msrp) +{ + struct avt_pdu_control_hdr *hdr = + (struct avt_pdu_control_hdr *)&avdecc->sd + .rx_buf[sizeof(struct ethhdr)]; + + if (avb_avdecc_listen(avdecc) > 0) { + switch (hdr->h.f.sub_type) { + case AVB_AVTP_SUBTYPE_ADP: + avb_avdecc_adp_respond_to_cmd(avdecc); + break; + + case AVB_AVTP_SUBTYPE_AECP: + avb_avdecc_aecp_respond_to_cmd(avdecc, msrp); + break; + + case AVB_AVTP_SUBTYPE_ACMP: + avb_avdecc_acmp_respond_to_cmd(avdecc); + break; + + default: + avb_log(AVB_KERN_INFO, + KERN_INFO + "avb_avdecc_listen_and_respond unknown sub_type: %d", + hdr->h.f.sub_type); + break; + } + } +} \ No newline at end of file diff --git a/sound/drivers/avb/avdecc.h b/sound/drivers/avb/avdecc.h new file mode 100644 index 0000000000000..3b39064bcbbc4 --- /dev/null +++ b/sound/drivers/avb/avdecc.h @@ -0,0 +1,454 @@ +#ifndef AVDECC_H +#define AVDECC_H + +#include "avb_util.h" +#include "msrp.h" + +#pragma pack(push, 1) + +struct avt_pdu_control_hdr { + union tch { + struct tcf { + u8 sub_type; + union tcb1 { + u8 sv; /* 1 bit stream valid indication */ + u8 version; /* 3 bits version */ + u8 msg_type; /* 4 bit ControlData/MessageType */ + } b1; + union tcb2 { + u8 valid_time; /* 5 bit Status/ValidTime */ + u8 data_len; /* First 3 bits of control data length */ + } b2; + u8 data_len; /* Last 8 bits of control data length */ + u8 stream_id[8]; /* Stream or entity id */ + } f; + u8 bytes[AVTP_PDU_COMMON_CONTROL_HEADER_LENGTH]; + } h; +}; + +struct maap_pdu { + union mtch { + struct mtcf { + u8 sub_type; + union mtcb1 { + u8 sv; /* 1 bit stream valid indication */ + u8 version; /* 3 bits version */ + u8 msg_type; /* 4 bit ControlData/MessageType */ + } b1; + union mtcb2 { + u8 maap_version; /* 5 bit Status/ValidTime */ + u8 data_len; /* First 3 bits of control data length */ + } b2; + u8 data_len; /* Last 8 bits of control data length */ + u8 stream_id[8]; /* Stream or entity id */ + } f; + u8 bytes[AVTP_PDU_COMMON_CONTROL_HEADER_LENGTH]; + } h; + u8 req_MAC[6]; + u16 req_count; + u8 conflict_MAC[6]; + u16 conflict_count; +}; + +struct avtp_stream_format { + u8 sub_type; + union sfb1 { + u8 res1; /* 4 bit reserved */ + u8 nsr; /* 4 bit Nominal sample rate */ + } b1; + u8 format; + u8 bit_depth; + u8 cpf; /* First 8 bits of channels per frame */ + union sfb5 { + u8 cpf; /* Last 2 bits of channels per frame */ + u8 spf; /* First 6 bits of samples per frame */ + } b5; + union sfb6 { + u8 spf; /* Last 4 bits of samples per frame */ + u8 res2; /* First 4 bits of reserved */ + } b6; + u8 res2; /* Last 8 bits of reserved */ +}; + +struct iec_stream_format { + u8 sub_type; + union isfb1 { + u8 sf; /* 1 bit stream format */ + u8 fmt; /* 6 bit fomrat */ + u8 r; /* 1 bit reserved */ + } b1; + union isfb2 { + u8 fdf_evt; /* 5 bits */ + u8 fdf_sfc; /* 3 bits */ + } b2; + u8 dbs; + union isfb4 { + u8 b; /* 1 bit */ + u8 nb; /* 1 bit */ + u8 res; /* 6 bits */ + } b4; + u8 label_iec_60958_cnt; + u8 label_mbla_cnt; + union isfb7 { + u8 label_midi_cnt; /* 4 bits */ + u8 label_smptecnt; /* 4 bits */ + } b7; +}; + +struct stream_format { + union fmt { + struct avtp_stream_format avtp; + struct iec_stream_format iec; + } fmt; +}; + +struct acm_pdu { + u8 ctrl_entity_id[8]; + u8 talker_entity_id[8]; + u8 listener_entity_id[8]; + u16 talker_unique_id; + u16 listener_unique_id; + u8 stream_dest_MAC[6]; + u16 connection_count; + u16 sequence_id; + u16 flags; + u16 stream_vlan_id; + u16 res; +}; + +struct aem_cmd { + u8 ctrl_entity_id[8]; + u16 seq_id; + u16 cmd_type; +}; + +struct acquire_ent_cmd { + struct aem_cmd hdr; + u32 flags; + u8 owner_id[8]; + u16 desc_type; + u16 desc_idx; +}; + +struct read_descp_cmd { + struct aem_cmd hdr; + u16 cfg_idx; + u16 res; + u16 desc_type; + u16 desc_idx; +}; + +struct set_stream_format_cmd { + struct aem_cmd hdr; + u16 desc_type; + u16 desc_idx; + struct stream_format fmt; +}; + +struct read_descp_res { + struct aem_cmd hdr; + u16 cfg_idx; + u16 res; +}; + +struct entity_descp { + u16 desc_type; + u16 desc_idx; + u8 entity_id[8]; + u8 entity_model_id[8]; + u32 entity_caps; + u16 talker_stream_sources; + u16 talker_caps; + u16 listener_stream_sinks; + u16 listener_caps; + u32 control_caps; + u32 avai_idx; + u8 association_id[8]; + u8 entity_name[64]; + u16 vendor_name_string; + u16 model_name_string; + u8 firmware_ver[64]; + u8 group_name[64]; + u8 serial_number[64]; + u16 cfg_count; + u16 curr_cfg; +}; + +struct config_descp_count { + u16 desc_type; + u16 desc_count; +}; + +struct config_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 descp_count; + u16 descpOff; + struct config_descp_count descps[AVB_AEM_MAX_DESCP_COUNT]; +}; + +struct audio_unit_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 clock_domain_idx; + u16 num_stream_ip; + u16 base_stream_ip; + u16 num_stream_op; + u16 base_stream_op; + u16 num_ext_ip; + u16 base_ext_ip; + u16 num_ext_op; + u16 base_ext_op; + u16 num_int_ip; + u16 base_int_ip; + u16 num_int_op; + u16 base_int_op; + u16 num_controls; + u16 base_control; + u16 num_signal_selector; + u16 base_signal_selector; + u16 num_mixers; + u16 base_mixer; + u16 num_matrices; + u16 base_matrix; + u16 num_splitters; + u16 base_splitter; + u16 num_combiners; + u16 base_combiner; + u16 numde_multiplexer; + u16 basede_multiplexer; + u16 num_multiplexer; + u16 base_multiplexer; + u16 num_transcoders; + u16 base_transcoder; + u16 num_control_blocks; + u16 base_control_block; + u32 current_sampling_rate; + u16 sampling_rates_offset; + u16 sampling_rates_count; + u32 sampling_rates[6]; +}; + +struct stream_port_descp { + u16 desc_type; + u16 desc_idx; + u16 clock_domain_idx; + u16 port_flags; + u16 num_controls; + u16 base_control; + u16 num_clusters; + u16 base_cluster; + u16 num_maps; + u16 base_map; +}; + +struct ext_port_descp { + u16 desc_type; + u16 desc_idx; + u16 clock_domain_idx; + u16 port_flags; + u16 num_controls; + u16 base_control; + u16 signal_type; + u16 signal_idx; + u16 signal_op; + u32 block_latency; + u16 jack_idx; +}; + +struct jack_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 jack_flags; + u16 jack_type; + u16 num_controls; + u16 base_control; +}; + +struct audio_cluster_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 signal_type; + u16 signal_idx; + u16 signal_op; + u32 path_latency; + u32 block_latency; + u16 num_channels; + u8 format; +}; + +struct aud_map_fmt { + u16 stream_idx; + u16 stream_channel; + u16 cluster_offset; + u16 cluster_channel; +}; + +struct audio_map_descp { + u16 desc_type; + u16 desc_idx; + u16 mapping_offset; + u16 num_mappings; + struct aud_map_fmt map[8]; +}; + +struct clock_source_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 clock_source_flags; + u16 clock_source_type; + u8 clock_source_id[8]; + u16 clock_source_loc_type; + u16 clock_source_loc_idx; +}; + +struct clock_domain_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 curr_clock_source; + u16 clock_sources_offset; + u16 clock_sources_count; + u16 clock_sources[3]; +}; + +struct avb_if_descp { + u16 desc_type; + u16 desc_idx; + u8 if_name[64]; + u16 localized_descp; + u8 mac_addr[6]; + u16 if_flags; + u8 clock_iden[8]; + u8 prio1; + u8 clock_class; + u16 off_scaled_log_var; + u8 clock_accu; + u8 prio2; + u8 domain_no; + u8 log_sync_int; + u8 log_anno_int; + u8 log_pdelay_int; + u16 port_no; +}; + +struct locale_descp { + u16 desc_type; + u16 desc_idx; + u8 locale_id[64]; + u16 num_strings; + u16 base_strings_idx; +}; + +struct strings_descp { + u16 desc_type; + u16 desc_idx; + u8 strings[7][64]; +}; + +struct stream_descp { + u16 desc_type; + u16 desc_idx; + u8 obj_name[64]; + u16 localized_descp; + u16 clock_domain_idx; + u16 stream_flags; + struct stream_format curr_fmt; + u16 fmts_off; + u16 fmts_count; + u8 bkp_talker1_entity_id[8]; + u16 bkp_talker1_unique_id; + u8 bkp_talker2_entity_id[8]; + u16 bkp_talker2_unique_id; + u8 bkp_talker3_entity_id[8]; + u16 bkp_talker3_unique_id; + u8 bkp_ip_talker_entity_id[8]; + u16 bkp_ip_talker_unique_id; + u16 avb_if_idx; + u32 buf_size; + struct stream_format supp_fmts[AVB_AEM_MAX_SUPP_FORMATS]; +}; + +struct get_counters_cmd { + struct aem_cmd hdr; + u16 desc_type; + u16 desc_idx; +}; + +struct counters_descp { + struct aem_cmd hdr; + u16 desc_type; + u16 desc_idx; + u32 counters_valid; + u32 counters[32]; +}; + +struct get_stream_info_cmd { + struct aem_cmd hdr; + u16 desc_type; + u16 desc_idx; +}; + +struct aem_stream_info { + struct aem_cmd hdr; + u16 desc_type; + u16 desc_idx; + u32 flags; + struct stream_format curr_fmt; + u8 stream_id[8]; + u32 msrp_accu_lat; + u8 stream_dest_MAC[6]; + u8 msrp_failure_code; + u8 res1; + u8 msrp_failure_bridge_id[8]; + u16 stream_vlan_id; + u16 res2; +}; + +struct adpdu { + u8 entity_model_id[8]; + u32 entity_caps; + u16 talker_stream_sources; + u16 talker_caps; + u16 listener_stream_sinks; + u16 listener_caps; + u32 control_caps; + u32 avai_idx; + u8 gptp_grand_master_id[8]; + u8 gptp_domain_number; + u8 res1[3]; + u16 iden_ctrl_idx; + u16 interface_idx; + u8 association_id[8]; + u32 res2; +}; + +#pragma pack(pop) + +struct avdecc { + bool initialized; + u8 acmp_tx_state; + u8 acmp_rx_state; + u32 adp_avai_idx; + u64 last_ADP_adv_jiffy; + struct socketdata sd; +}; + +bool avb_avdecc_init(struct avdecc *avdecc); +void avb_adp_discover(struct avdecc *avdecc); +void avb_adp_advertise(struct avdecc *avdecc); +void avb_maap_announce(struct avdecc *avdecc); +void avb_avdecc_listen_and_respond(struct avdecc *avdecc, struct msrp *msrp); + +#endif \ No newline at end of file diff --git a/sound/drivers/avb/avtp.c b/sound/drivers/avb/avtp.c new file mode 100644 index 0000000000000..89f3b753c34b0 --- /dev/null +++ b/sound/drivers/avb/avtp.c @@ -0,0 +1,105 @@ +#include "avtp.h" + +static int avb_get_avtp_aaf_format(int rt_format) +{ + int format = AVB_AVTP_AAF_FORMAT_USER_SP; + + if ((rt_format == SNDRV_PCM_FORMAT_FLOAT_LE) || + (rt_format == SNDRV_PCM_FORMAT_FLOAT_BE)) + format = AVB_AVTP_AAF_FORMAT_32_BIT_FLOAT; + else if ((rt_format == SNDRV_PCM_FORMAT_S32_LE) || + (rt_format == SNDRV_PCM_FORMAT_S32_BE) || + (rt_format == SNDRV_PCM_FORMAT_U32_LE) || + (rt_format == SNDRV_PCM_FORMAT_U32_BE)) + format = AVB_AVTP_AAF_FORMAT_32_BIT_INT; + else if ((rt_format == SNDRV_PCM_FORMAT_S24_LE) || + (rt_format == SNDRV_PCM_FORMAT_S24_BE) || + (rt_format == SNDRV_PCM_FORMAT_U24_LE) || + (rt_format == SNDRV_PCM_FORMAT_U24_BE)) + format = AVB_AVTP_AAF_FORMAT_24_bit_INT; + else if ((rt_format == SNDRV_PCM_FORMAT_S16_LE) || + (rt_format == SNDRV_PCM_FORMAT_S16_BE) || + (rt_format == SNDRV_PCM_FORMAT_U16_LE) || + (rt_format == SNDRV_PCM_FORMAT_U16_BE)) + format = AVB_AVTP_AAF_FORMAT_16_BIT_INT; + else + format = AVB_AVTP_AAF_FORMAT_USER_SP; + + return format; +} + +static int avb_get_avtp_aaf_nsr(int sample_rate) +{ + int nsr = AVB_AVTP_AAF_NSR_USER_SP; + + if (sample_rate == 8000) + nsr = AVB_AVTP_AAF_NSR_8_KHZ; + else if (sample_rate == 16000) + nsr = AVB_AVTP_AAF_NSR_16_KHZ; + else if (sample_rate == 32000) + nsr = AVB_AVTP_AAF_NSR_32_KHZ; + else if (sample_rate == 44100) + nsr = AVB_AVTP_AAF_NSR_44_1_KHZ; + else if (sample_rate == 48000) + nsr = AVB_AVTP_AAF_NSR_48_KHZ; + else if (sample_rate == 88200) + nsr = AVB_AVTP_AAF_NSR_88_2_KHZ; + else if (sample_rate == 96000) + nsr = AVB_AVTP_AAF_NSR_96_KHZ; + else if (sample_rate == 176400) + nsr = AVB_AVTP_AAF_NSR_176_4_KHZ; + else if (sample_rate == 192000) + nsr = AVB_AVTP_AAF_NSR_192_KHZ; + else if (sample_rate == 24000) + nsr = AVB_AVTP_AAF_NSR_24_KHZ; + else + nsr = AVB_AVTP_AAF_NSR_USER_SP; + + return nsr; +} + +void avb_avtp_aaf_header_init(char *buf, struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *hw_params) +{ + struct avb_card *avb_card = snd_pcm_substream_chip(substream); + struct ethhdr *eh = (struct ethhdr *)&buf[0]; + struct avt_pdu_aaf_pcm_hdr *hdr = + (struct avt_pdu_aaf_pcm_hdr *)&buf[sizeof(struct ethhdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_avtp_aaf_header_init"); + + memset(buf, 0, AVB_MAX_ETH_FRAME_SIZE); + + eh->h_dest[0] = avb_card->sd.destmac[0]; + eh->h_dest[1] = avb_card->sd.destmac[1]; + eh->h_dest[2] = avb_card->sd.destmac[2]; + eh->h_dest[3] = avb_card->sd.destmac[3]; + eh->h_dest[4] = avb_card->sd.destmac[4]; + eh->h_dest[5] = avb_card->sd.destmac[5]; + eh->h_source[0] = avb_card->sd.srcmac[0]; + eh->h_source[1] = avb_card->sd.srcmac[1]; + eh->h_source[2] = avb_card->sd.srcmac[2]; + eh->h_source[3] = avb_card->sd.srcmac[3]; + eh->h_source[4] = avb_card->sd.srcmac[4]; + eh->h_source[5] = avb_card->sd.srcmac[5]; + + eh->h_proto = htons(avb_card->sd.type); + + hdr->h.f.sub_type = AVB_AVTP_SUBTYPE_AAF; + AVB_AVTP_AAF_HDR_SET_SV(hdr, 1); + AVB_AVTP_AAF_HDR_SET_VER(hdr, AVB_AVTP_AAF_VERSION); + AVB_AVTP_AAF_HDR_SET_MR(hdr, 0); + AVB_AVTP_AAF_HDR_SET_TSV(hdr, 1); + hdr->h.f.seq_no = 0; + AVB_AVTP_AAF_HDR_SET_TU(hdr, 0); + hdr->h.f.stream_id = 0; + hdr->h.f.avtp_ts = 0; + hdr->h.f.format = avb_get_avtp_aaf_format(substream->runtime->format); + AVB_AVTP_AAF_HDR_SET_NSR(hdr, + avb_get_avtp_aaf_nsr(params_rate(hw_params))); + AVB_AVTP_AAF_HDR_SET_CPF(hdr, params_channels(hw_params)); + hdr->h.f.bit_depth = substream->runtime->sample_bits; + hdr->h.f.stream_data_len = 0; + AVB_AVTP_AAF_HDR_SET_SP(hdr, 1); + AVB_AVTP_AAF_HDR_SET_EVT(hdr, 0); +} diff --git a/sound/drivers/avb/avtp.h b/sound/drivers/avb/avtp.h new file mode 100644 index 0000000000000..f55e0bcc37aeb --- /dev/null +++ b/sound/drivers/avb/avtp.h @@ -0,0 +1,51 @@ +#ifndef AVTP_H +#define AVTP_H + +#include "avb_util.h" + +#pragma pack(push, 1) + +struct avt_pdu_aaf_pcm_hdr { + union th { + struct tf { + u8 sub_type; + union tb1 { + u8 sv; /* 1 bit stream valid indication */ + u8 version; /* 3 bits version */ + u8 mr; /* 1 bit media clock restart */ + u8 rsv; /* 2 bits reserved */ + u8 ts_valid; /* 1 bit timestamp valid */ + } b1; + u8 seq_no; + union tb2 { + u8 rsv; /* 7 bit reserved data */ + u8 tu; /* 1 bit timestamp uncertain */ + } b2; + u64 stream_id; + u32 avtp_ts; + u8 format; + union tfsd1 { + u8 nsr; /* 4 bits nominal sample rate */ + u8 rsv; /* 2 bits reserved data */ + u8 cpf; /* first 2 bits of channels per frame */ + } fsd1; + u8 cpf; /* last 8 bits of channels per frame */ + u8 bit_depth; + u16 stream_data_len; + union tfsd2 { + u8 rsv; /* 3 bits reserved data */ + u8 sp; /* 1 bit sparse timestamp */ + u8 evt; /* 4 bits event data */ + } fsd2; + u8 rsv; + } f; + u8 bytes[AVTP_PDU_COMMON_STREAM_HEADER_LENGTH]; + } h; +}; + +#pragma pack(pop) + +void avb_avtp_aaf_header_init(char *buf, struct snd_pcm_substream *substream, + struct snd_pcm_hw_params *hw_params); + +#endif \ No newline at end of file diff --git a/sound/drivers/avb/msrp.c b/sound/drivers/avb/msrp.c new file mode 100644 index 0000000000000..e3f053dd0d255 --- /dev/null +++ b/sound/drivers/avb/msrp.c @@ -0,0 +1,391 @@ +#include "msrp.h" + +bool avb_msrp_init(struct msrp *msrp) +{ + avb_log(AVB_KERN_INFO, KERN_INFO "avb_msrp_init"); + + msrp->rx_state = MSRP_DECLARATION_STATE_NONE; + msrp->tx_state = MSRP_DECLARATION_STATE_NONE; + + msrp->sd.type = ETH_MSRP; + msrp->sd.destmac[0] = 0x01; + msrp->sd.destmac[1] = 0x80; + msrp->sd.destmac[2] = 0xC2; + msrp->sd.destmac[3] = 0x00; + msrp->sd.destmac[4] = 0x00; + msrp->sd.destmac[5] = 0x0E; + + return avb_socket_init(&msrp->sd, 1000); +} + +void avb_msrp_domaindeclarations(struct msrp *msrp) +{ + int tx_size = 0; + int err = 0; + struct ethhdr *eh = (struct ethhdr *)&msrp->sd.tx_buf[0]; + struct domain_msrp_du *pdu = + (struct domain_msrp_du *)&msrp->sd.tx_buf[sizeof(struct ethhdr)]; + + avb_log(AVB_KERN_INFO, KERN_INFO "avb_msrp_domaindeclarations"); + + /* Initialize it */ + memset(msrp->sd.tx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + + /* Fill in the Ethernet header */ + eh->h_dest[0] = 0x01; + eh->h_dest[1] = 0x80; + eh->h_dest[2] = 0xC2; + eh->h_dest[3] = 0x00; + eh->h_dest[4] = 0x00; + eh->h_dest[5] = 0x0E; + eh->h_source[0] = msrp->sd.srcmac[0]; + eh->h_source[1] = msrp->sd.srcmac[1]; + eh->h_source[2] = msrp->sd.srcmac[2]; + eh->h_source[3] = msrp->sd.srcmac[3]; + eh->h_source[4] = msrp->sd.srcmac[4]; + eh->h_source[5] = msrp->sd.srcmac[5]; + + /* Fill in Ethertype field */ + eh->h_proto = htons(msrp->sd.type); + + pdu->protocol_version = 0; + pdu->msg.attribute_type = MSRP_ATTRIBUTE_TYPE_DOMAIN_VECTOR; + pdu->msg.attribute_len = MSRP_ATTRIBUTE_LEN_DOMAIN_VECTOR; + pdu->msg.attribute_list_len = + htons(sizeof(struct domain_vector_attribute) + 2); + pdu->msg.attibute_list.hdr.number_of_values = htons(1); + pdu->msg.attibute_list.val.sr_class_id = 2; + pdu->msg.attibute_list.val.sr_class_prio = 3; + pdu->msg.attibute_list.val.sr_class_VID = htons(2); + pdu->msg.attibute_list.vector[0] = + MSRP_THREE_PACK(MSRP_ATTRIBUTE_EVENT_JOININ, 0, 0); + pdu->msg.end_marker = 0; + pdu->end_marker = 0; + + tx_size = sizeof(struct ethhdr) + sizeof(struct domain_msrp_du); + + msrp->sd.tx_iov.iov_base = msrp->sd.tx_buf; + msrp->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&msrp->sd.tx_msg_hdr.msg_iter, WRITE, &msrp->sd.tx_iov, 1, + tx_size); + + if ((err = sock_sendmsg(msrp->sd.sock, &msrp->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_domaindeclarations Socket transmission fails %d \n", + err); + return; + } +} + +void avb_msrp_talkerdeclarations(struct msrp *msrp, bool join, int state) +{ + int tx_size = 0; + int err = 0; + struct ethhdr *eh = (struct ethhdr *)&msrp->sd.tx_buf[0]; + struct talker_msrp_du *pdu = + (struct talker_msrp_du *)&msrp->sd.tx_buf[sizeof(struct ethhdr)]; + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_msrp_talkerdeclarations join: %d, state: %d", + join, state); + + /* Initialize it */ + memset(msrp->sd.tx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + + /* Fill in the Ethernet header */ + eh->h_dest[0] = 0x01; + eh->h_dest[1] = 0x80; + eh->h_dest[2] = 0xC2; + eh->h_dest[3] = 0x00; + eh->h_dest[4] = 0x00; + eh->h_dest[5] = 0x0E; + eh->h_source[0] = msrp->sd.srcmac[0]; + eh->h_source[1] = msrp->sd.srcmac[1]; + eh->h_source[2] = msrp->sd.srcmac[2]; + eh->h_source[3] = msrp->sd.srcmac[3]; + eh->h_source[4] = msrp->sd.srcmac[4]; + eh->h_source[5] = msrp->sd.srcmac[5]; + + /* Fill in Ethertype field */ + eh->h_proto = htons(msrp->sd.type); + + pdu->protocol_version = 0; + pdu->msg.attribute_type = MSRP_ATTRIBUTE_TYPE_TALKER_ADVERTISE_VECTOR; + pdu->msg.attribute_len = MSRP_ATTRIBUTE_LEN_TALKER_ADVERTISE_VECTOR; + pdu->msg.attribute_list_len = + htons(sizeof(struct talker_vector_attribute)); + + if (state != MSRP_DECLARATION_STATE_UNKNOWN) { + pdu->msg.attibute_list.hdr.number_of_values = htons(1); + pdu->msg.attibute_list.val.stream_id[0] = msrp->sd.srcmac[0]; + pdu->msg.attibute_list.val.stream_id[1] = msrp->sd.srcmac[1]; + pdu->msg.attibute_list.val.stream_id[2] = msrp->sd.srcmac[2]; + pdu->msg.attibute_list.val.stream_id[3] = msrp->sd.srcmac[3]; + pdu->msg.attibute_list.val.stream_id[4] = msrp->sd.srcmac[4]; + pdu->msg.attibute_list.val.stream_id[5] = msrp->sd.srcmac[5]; + pdu->msg.attibute_list.val.stream_id[6] = 0; + pdu->msg.attibute_list.val.stream_id[7] = 1; + + pdu->msg.attibute_list.val.data_frame_params[0] = + msrp->sd.destmac[0]; + pdu->msg.attibute_list.val.data_frame_params[1] = + msrp->sd.destmac[1]; + pdu->msg.attibute_list.val.data_frame_params[2] = + msrp->sd.destmac[2]; + pdu->msg.attibute_list.val.data_frame_params[3] = + msrp->sd.destmac[3]; + pdu->msg.attibute_list.val.data_frame_params[4] = + msrp->sd.destmac[4]; + pdu->msg.attibute_list.val.data_frame_params[5] = + msrp->sd.destmac[5]; + pdu->msg.attibute_list.val.data_frame_params[6] = 0; + pdu->msg.attibute_list.val.data_frame_params[7] = 2; + + pdu->msg.attibute_list.val.max_frame_size = + htons(MSRP_MAX_FRAME_SIZE_48KHZ_AUDIO); + pdu->msg.attibute_list.val.max_interval_frames = + htons(MSRP_MAX_INTERVAL_FRAME_48KHZ_AUDIO); + pdu->msg.attibute_list.val.priority_and_rank = 0; + pdu->msg.attibute_list.val.accumalated_latency = htonl(0); + + pdu->msg.attibute_list.vector[0] = MSRP_THREE_PACK( + ((join == true) ? (MSRP_ATTRIBUTE_EVENT_JOININ) : + (MSRP_ATTRIBUTE_EVENT_LEAVE)), + 0, 0); + } else { + pdu->msg.attibute_list.vector[0] = + MSRP_THREE_PACK(MSRP_ATTRIBUTE_EVENT_JOINMT, 0, 0); + } + + pdu->msg.end_marker = 0; + pdu->end_marker = 0; + + tx_size = sizeof(struct ethhdr) + sizeof(struct talker_msrp_du); + + msrp->sd.tx_iov.iov_base = msrp->sd.tx_buf; + msrp->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&msrp->sd.tx_msg_hdr.msg_iter, WRITE, &msrp->sd.tx_iov, 1, + tx_size); + + if ((err = sock_sendmsg(msrp->sd.sock, &msrp->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_talkerdeclarations Socket transmission fails %d \n", + err); + return; + } +} + +void avb_msrp_listenerdeclarations(struct msrp *msrp, bool join, int state) +{ + int tx_size = 0; + int err = 0; + struct ethhdr *eh = (struct ethhdr *)&msrp->sd.tx_buf[0]; + struct listner_msrp_du *pdu = (struct listner_msrp_du *)&msrp->sd + .tx_buf[sizeof(struct ethhdr)]; + + avb_log(AVB_KERN_INFO, + KERN_INFO "avb_msrp_listenerdeclarations join: %d, state: %d", + join, state); + + /* Initialize it */ + memset(msrp->sd.tx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + + /* Fill in the Ethernet header */ + eh->h_dest[0] = 0x01; + eh->h_dest[1] = 0x80; + eh->h_dest[2] = 0xC2; + eh->h_dest[3] = 0x00; + eh->h_dest[4] = 0x00; + eh->h_dest[5] = 0x0E; + eh->h_source[0] = msrp->sd.srcmac[0]; + eh->h_source[1] = msrp->sd.srcmac[1]; + eh->h_source[2] = msrp->sd.srcmac[2]; + eh->h_source[3] = msrp->sd.srcmac[3]; + eh->h_source[4] = msrp->sd.srcmac[4]; + eh->h_source[5] = msrp->sd.srcmac[5]; + + /* Fill in Ethertype field */ + eh->h_proto = htons(msrp->sd.type); + + pdu->protocol_version = 0; + pdu->msg.attribute_type = MSRP_ATTRIBUTE_TYPE_LISTENER_VECTOR; + pdu->msg.attribute_len = MSRP_ATTRIBUTE_LEN_LISTENER_VECTOR; + pdu->msg.attribute_list_len = + htons(sizeof(struct listner_vector_attribute)); + + if (state != MSRP_DECLARATION_STATE_UNKNOWN) { + pdu->msg.attibute_list.hdr.number_of_values = htons(1); + pdu->msg.attibute_list.val.stream_id[0] = msrp->stream_id[0]; + pdu->msg.attibute_list.val.stream_id[1] = msrp->stream_id[1]; + pdu->msg.attibute_list.val.stream_id[2] = msrp->stream_id[2]; + pdu->msg.attibute_list.val.stream_id[3] = msrp->stream_id[3]; + pdu->msg.attibute_list.val.stream_id[4] = msrp->stream_id[4]; + pdu->msg.attibute_list.val.stream_id[5] = msrp->stream_id[5]; + pdu->msg.attibute_list.val.stream_id[6] = msrp->stream_id[6]; + pdu->msg.attibute_list.val.stream_id[7] = msrp->stream_id[7]; + + pdu->msg.attibute_list.vector[0] = MSRP_THREE_PACK( + ((join == true) ? (MSRP_ATTRIBUTE_EVENT_JOININ) : + (MSRP_ATTRIBUTE_EVENT_LEAVE)), + 0, 0); + pdu->msg.attibute_list.vector[1] = + MSRP_FOUR_PACK(state, 0, 0, 0); + } else { + pdu->msg.attibute_list.vector[0] = + MSRP_THREE_PACK(MSRP_ATTRIBUTE_EVENT_JOINMT, 0, 0); + pdu->msg.attibute_list.vector[1] = + MSRP_FOUR_PACK(MSRP_DECLARATION_STATE_READY, 0, 0, 0); + } + + pdu->msg.end_marker = 0; + pdu->end_marker = 0; + + tx_size = sizeof(struct ethhdr) + sizeof(struct listner_msrp_du); + + msrp->sd.tx_iov.iov_base = msrp->sd.tx_buf; + msrp->sd.tx_iov.iov_len = tx_size; + iov_iter_init(&msrp->sd.tx_msg_hdr.msg_iter, WRITE, &msrp->sd.tx_iov, 1, + tx_size); + + if ((err = sock_sendmsg(msrp->sd.sock, &msrp->sd.tx_msg_hdr)) <= 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_listenerdeclarations Socket transmission fails %d \n", + err); + return; + } +} + +static void avb_msrp_evaluate_talker_advertisement(struct msrp *msrp) +{ + struct talker_msrp_du *tpdu = + (struct talker_msrp_du *)&msrp->sd.rx_buf[sizeof(struct ethhdr)]; + + int leave_all = + ntohs(tpdu->msg.attibute_list.hdr.number_of_values) & 0x2000; + int evt = MSRP_THREE_PACK_GET_A(tpdu->msg.attibute_list.vector[0]); + + if (leave_all != 0) { + msrp->tx_state = MSRP_DECLARATION_STATE_UNKNOWN; + } else { + if (tpdu->msg.attribute_type == + MSRP_ATTRIBUTE_TYPE_TALKER_ADVERTISE_VECTOR) { + if ((evt == MSRP_ATTRIBUTE_EVENT_JOINMT) || + (evt == MSRP_ATTRIBUTE_EVENT_MT) || + (evt == MSRP_ATTRIBUTE_EVENT_LEAVE)) { + msrp->rx_state = MSRP_DECLARATION_STATE_NONE; + memset(&msrp->stream_id[0], 0, 8); + } else { + msrp->rx_state = MSRP_DECLARATION_STATE_READY; + } + } else { + msrp->rx_state = MSRP_DECLARATION_STATE_ASKING_FAILED; + } + + memcpy(&msrp->stream_id[0], + &tpdu->msg.attibute_list.val.stream_id[0], 8); + } +} + +static void avb_msrp_evaluate_listener_advertisement(struct msrp *msrp) +{ + struct listner_msrp_du *pdu = (struct listner_msrp_du *)&msrp->sd + .rx_buf[sizeof(struct ethhdr)]; + + int leave_all = + ntohs(pdu->msg.attibute_list.hdr.number_of_values) & 0x2000; + int evt = MSRP_THREE_PACK_GET_A(pdu->msg.attibute_list.vector[0]); + + if (leave_all != 0) { + msrp->rx_state = MSRP_DECLARATION_STATE_UNKNOWN; + } else { + if ((evt == MSRP_ATTRIBUTE_EVENT_JOINMT) || + (evt == MSRP_ATTRIBUTE_EVENT_MT) || + (evt == MSRP_ATTRIBUTE_EVENT_LEAVE)) { + msrp->tx_state = MSRP_DECLARATION_STATE_NONE; + } else { + if ((pdu->msg.attibute_list.val.stream_id[0] == + msrp->sd.srcmac[0]) && + (pdu->msg.attibute_list.val.stream_id[1] == + msrp->sd.srcmac[1]) && + (pdu->msg.attibute_list.val.stream_id[2] == + msrp->sd.srcmac[2]) && + (pdu->msg.attibute_list.val.stream_id[3] == + msrp->sd.srcmac[3]) && + (pdu->msg.attibute_list.val.stream_id[4] == + msrp->sd.srcmac[4]) && + (pdu->msg.attibute_list.val.stream_id[5] == + msrp->sd.srcmac[5]) && + (pdu->msg.attibute_list.val.stream_id[6] == 0) && + (pdu->msg.attibute_list.val.stream_id[7] == 1)) { + msrp->tx_state = MSRP_DECLARATION_STATE_READY; + } + } + } +} + +int avb_msrp_listen(struct msrp *msrp) +{ + int err = 0; + mm_segment_t oldfs; + struct listner_msrp_du *tpdu = (struct listner_msrp_du *)&msrp->sd + .rx_buf[sizeof(struct ethhdr)]; + struct kvec vec; + + memset(msrp->sd.rx_buf, 0, AVB_MAX_ETH_FRAME_SIZE); + msrp->sd.rx_iov.iov_base = msrp->sd.rx_buf; + msrp->sd.rx_iov.iov_len = AVB_MAX_ETH_FRAME_SIZE; + iov_iter_init(&msrp->sd.rx_msg_hdr.msg_iter, READ, &msrp->sd.rx_iov, 1, + AVB_MAX_ETH_FRAME_SIZE); + + vec.iov_base = msrp->sd.rx_buf; + vec.iov_len = AVB_MAX_ETH_FRAME_SIZE; + + oldfs = get_fs(); + set_fs(KERNEL_DS); + err = kernel_recvmsg(msrp->sd.sock, &msrp->sd.rx_msg_hdr, &vec, 1, + AVB_MAX_ETH_FRAME_SIZE, MSG_DONTWAIT); + set_fs(oldfs); + + if (err <= 0) { + if (err != -11) + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_listen Socket reception res %d \n", + err); + } else { + if (tpdu->protocol_version != 0) { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_listen unknown protocol_version %d \n", + tpdu->protocol_version); + } else { + if ((tpdu->msg.attribute_type == + MSRP_ATTRIBUTE_TYPE_TALKER_ADVERTISE_VECTOR) || + (tpdu->msg.attribute_type == + MSRP_ATTRIBUTE_TYPE_TALKER_FAILED_VECTOR)) { + avb_msrp_evaluate_talker_advertisement(msrp); + } else if (tpdu->msg.attribute_type == + MSRP_ATTRIBUTE_TYPE_LISTENER_VECTOR) { + avb_msrp_evaluate_listener_advertisement(msrp); + } else if (tpdu->msg.attribute_type == + MSRP_ATTRIBUTE_TYPE_DOMAIN_VECTOR) { + } else { + avb_log(AVB_KERN_WARN, + KERN_WARNING + "avb_msrp_listen unknown attribute type %d \n", + tpdu->msg.attribute_type); + } + + avb_log(AVB_KERN_NOT, + KERN_NOTICE + "avb_msrp_listen: rxType: %d, rx_state: %d, tx_state: %d", + tpdu->msg.attribute_type, msrp->rx_state, + msrp->tx_state); + } + } + + return err; +} \ No newline at end of file diff --git a/sound/drivers/avb/msrp.h b/sound/drivers/avb/msrp.h new file mode 100644 index 0000000000000..735caed5f0f81 --- /dev/null +++ b/sound/drivers/avb/msrp.h @@ -0,0 +1,119 @@ +#ifndef MSRP_H +#define MSRP_H + +#include "avb_util.h" + +#pragma pack(push, 1) + +struct domain_msrp_first_value { + u8 sr_class_id; + u8 sr_class_prio; + u16 sr_class_VID; +}; + +struct listener_msrp_first_value { + u8 stream_id[8]; +}; + +struct talker_msrp_first_value { + u8 stream_id[8]; + u8 data_frame_params[8]; + u16 max_frame_size; + u16 max_interval_frames; + u8 priority_and_rank; + u32 accumalated_latency; +}; + +struct bridge_msrp_firstvalue { + u8 stream_id[8]; + u8 data_frame_params[8]; + u16 max_frame_size; + u16 max_interval_frames; + u8 priority_and_rank; + u32 accumalated_latency; + u8 bridge_id[8]; + u8 failure_reason; +}; + +struct vector_header { + u16 number_of_values; +}; + +struct domain_vector_attribute { + struct vector_header hdr; + struct domain_msrp_first_value val; + u8 vector[2]; +}; + +struct listner_vector_attribute { + struct vector_header hdr; + struct listener_msrp_first_value val; + u8 vector[2]; +}; + +struct talker_vector_attribute { + struct vector_header hdr; + struct talker_msrp_first_value val; + u8 vector[1]; +}; + +struct domain_mrp_msg { + u8 attribute_type; + u8 attribute_len; + u16 attribute_list_len; + struct domain_vector_attribute attibute_list; + u16 end_marker; +}; + +struct listner_mrp_msg { + u8 attribute_type; + u8 attribute_len; + u16 attribute_list_len; + struct listner_vector_attribute attibute_list; + u16 end_marker; +}; + +struct talker_mrp_msg { + u8 attribute_type; + u8 attribute_len; + u16 attribute_list_len; + struct talker_vector_attribute attibute_list; + u16 end_marker; +}; + +struct domain_msrp_du { + u8 protocol_version; + struct domain_mrp_msg msg; + u16 end_marker; +}; + +struct listner_msrp_du { + u8 protocol_version; + struct listner_mrp_msg msg; + u16 end_marker; +}; + +struct talker_msrp_du { + u8 protocol_version; + struct talker_mrp_msg msg; + u16 end_marker; +}; + +#pragma pack(pop) + +struct msrp { + bool initialized; + bool started; + int rx_state; + int tx_state; + struct socketdata sd; + u8 stream_id[8]; +}; + +bool avb_msrp_init(struct msrp *msrp); +void avb_msrp_domaindeclarations(struct msrp *msrp); +void avb_msrp_talkerdeclarations(struct msrp *msrp, bool join, int state); +void avb_msrp_listenerdeclarations(struct msrp *msrp, bool join, int state); +int avb_msrp_listen(struct msrp *msrp); + +#endif \ No newline at end of file