ALSA: firewire-motu: enable to read transaction cache via hwdep interface
authorTakashi Sakamoto <o-takashi@sakamocchi.jp>
Wed, 22 Mar 2017 12:30:25 +0000 (21:30 +0900)
committerTakashi Iwai <tiwai@suse.de>
Tue, 28 Mar 2017 10:34:06 +0000 (12:34 +0200)
MOTU FireWire series can transfer messages to registered address. These
messages are transferred for the status of internal clock synchronization
just after starting streams.

When the synchronization is stable, it's 0x01ffffff. Else, it's 0x05ffffff.

This commit adds a functionality for user space applications to receive
content of the message.

Signed-off-by: Takashi Sakamoto <o-takashi@sakamocchi.jp>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
include/uapi/sound/firewire.h
sound/firewire/motu/motu-hwdep.c
sound/firewire/motu/motu-transaction.c

index 59c6d81f53644e43a486cb1a3f894de4602c88bf..29afc5eab42dc6afd5445fc85140b836b04d7bbf 100644 (file)
@@ -10,6 +10,7 @@
 #define SNDRV_FIREWIRE_EVENT_DICE_NOTIFICATION 0xd1ce004e
 #define SNDRV_FIREWIRE_EVENT_EFW_RESPONSE      0x4e617475
 #define SNDRV_FIREWIRE_EVENT_DIGI00X_MESSAGE   0x746e736c
+#define SNDRV_FIREWIRE_EVENT_MOTU_NOTIFICATION 0x64776479
 
 struct snd_firewire_event_common {
        unsigned int type; /* SNDRV_FIREWIRE_EVENT_xxx */
@@ -46,12 +47,18 @@ struct snd_firewire_event_digi00x_message {
        __u32 message;  /* Digi00x-specific message */
 };
 
+struct snd_firewire_event_motu_notification {
+       unsigned int type;
+       __u32 message;  /* MOTU-specific bits. */
+};
+
 union snd_firewire_event {
        struct snd_firewire_event_common            common;
        struct snd_firewire_event_lock_status       lock_status;
        struct snd_firewire_event_dice_notification dice_notification;
        struct snd_firewire_event_efw_response      efw_response;
        struct snd_firewire_event_digi00x_message   digi00x_message;
+       struct snd_firewire_event_motu_notification motu_notification;
 };
 
 
index e795a5219a21f6f4ea58445eb73b9c78d4e78335..b87ccb69d597978c75f89157defb8406f3c681b3 100644 (file)
@@ -26,7 +26,7 @@ static long hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
 
        spin_lock_irq(&motu->lock);
 
-       while (!motu->dev_lock_changed) {
+       while (!motu->dev_lock_changed && motu->msg == 0) {
                prepare_to_wait(&motu->hwdep_wait, &wait, TASK_INTERRUPTIBLE);
                spin_unlock_irq(&motu->lock);
                schedule();
@@ -43,6 +43,12 @@ static long hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
                motu->dev_lock_changed = false;
 
                count = min_t(long, count, sizeof(event.lock_status));
+       } else {
+               event.motu_notification.type = SNDRV_FIREWIRE_EVENT_MOTU_NOTIFICATION;
+               event.motu_notification.message = motu->msg;
+               motu->msg = 0;
+
+               count = min_t(long, count, sizeof(event.motu_notification));
        }
 
        spin_unlock_irq(&motu->lock);
@@ -62,7 +68,7 @@ static unsigned int hwdep_poll(struct snd_hwdep *hwdep, struct file *file,
        poll_wait(file, &motu->hwdep_wait, wait);
 
        spin_lock_irq(&motu->lock);
-       if (motu->dev_lock_changed)
+       if (motu->dev_lock_changed || motu->msg)
                events = POLLIN | POLLRDNORM;
        else
                events = 0;
index 416dd98338968291ffdf1e44cba1ed89eaee794d..7fc30091e0deef80656382ce4c2cdf6423dbc816 100644 (file)
@@ -50,7 +50,27 @@ static void handle_message(struct fw_card *card, struct fw_request *request,
                           int generation, unsigned long long offset,
                           void *data, size_t length, void *callback_data)
 {
+       struct snd_motu *motu = callback_data;
+       __be32 *buf = (__be32 *)data;
+       unsigned long flags;
+
+       if (tcode != TCODE_WRITE_QUADLET_REQUEST) {
+               fw_send_response(card, request, RCODE_COMPLETE);
+               return;
+       }
+
+       if (offset != motu->async_handler.offset || length != 4) {
+               fw_send_response(card, request, RCODE_ADDRESS_ERROR);
+               return;
+       }
+
+       spin_lock_irqsave(&motu->lock, flags);
+       motu->msg = be32_to_cpu(*buf);
+       spin_unlock_irqrestore(&motu->lock, flags);
+
        fw_send_response(card, request, RCODE_COMPLETE);
+
+       wake_up(&motu->hwdep_wait);
 }
 
 int snd_motu_transaction_reregister(struct snd_motu *motu)