motu-transaction.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. /*
  2. * motu-transaction.c - a part of driver for MOTU FireWire series
  3. *
  4. * Copyright (c) 2015-2017 Takashi Sakamoto <o-takashi@sakamocchi.jp>
  5. *
  6. * Licensed under the terms of the GNU General Public License, version 2.
  7. */
  8. #include "motu.h"
  9. #define SND_MOTU_ADDR_BASE 0xfffff0000000ULL
  10. #define ASYNC_ADDR_HI 0x0b04
  11. #define ASYNC_ADDR_LO 0x0b08
  12. int snd_motu_transaction_read(struct snd_motu *motu, u32 offset, __be32 *reg,
  13. size_t size)
  14. {
  15. int tcode;
  16. if (size % sizeof(__be32) > 0 || size <= 0)
  17. return -EINVAL;
  18. if (size == sizeof(__be32))
  19. tcode = TCODE_READ_QUADLET_REQUEST;
  20. else
  21. tcode = TCODE_READ_BLOCK_REQUEST;
  22. return snd_fw_transaction(motu->unit, tcode,
  23. SND_MOTU_ADDR_BASE + offset, reg, size, 0);
  24. }
  25. int snd_motu_transaction_write(struct snd_motu *motu, u32 offset, __be32 *reg,
  26. size_t size)
  27. {
  28. int tcode;
  29. if (size % sizeof(__be32) > 0 || size <= 0)
  30. return -EINVAL;
  31. if (size == sizeof(__be32))
  32. tcode = TCODE_WRITE_QUADLET_REQUEST;
  33. else
  34. tcode = TCODE_WRITE_BLOCK_REQUEST;
  35. return snd_fw_transaction(motu->unit, tcode,
  36. SND_MOTU_ADDR_BASE + offset, reg, size, 0);
  37. }
  38. static void handle_message(struct fw_card *card, struct fw_request *request,
  39. int tcode, int destination, int source,
  40. int generation, unsigned long long offset,
  41. void *data, size_t length, void *callback_data)
  42. {
  43. struct snd_motu *motu = callback_data;
  44. __be32 *buf = (__be32 *)data;
  45. unsigned long flags;
  46. if (tcode != TCODE_WRITE_QUADLET_REQUEST) {
  47. fw_send_response(card, request, RCODE_COMPLETE);
  48. return;
  49. }
  50. if (offset != motu->async_handler.offset || length != 4) {
  51. fw_send_response(card, request, RCODE_ADDRESS_ERROR);
  52. return;
  53. }
  54. spin_lock_irqsave(&motu->lock, flags);
  55. motu->msg = be32_to_cpu(*buf);
  56. spin_unlock_irqrestore(&motu->lock, flags);
  57. fw_send_response(card, request, RCODE_COMPLETE);
  58. wake_up(&motu->hwdep_wait);
  59. }
  60. int snd_motu_transaction_reregister(struct snd_motu *motu)
  61. {
  62. struct fw_device *device = fw_parent_device(motu->unit);
  63. __be32 data;
  64. int err;
  65. if (motu->async_handler.callback_data == NULL)
  66. return -EINVAL;
  67. /* Register messaging address. Block transaction is not allowed. */
  68. data = cpu_to_be32((device->card->node_id << 16) |
  69. (motu->async_handler.offset >> 32));
  70. err = snd_motu_transaction_write(motu, ASYNC_ADDR_HI, &data,
  71. sizeof(data));
  72. if (err < 0)
  73. return err;
  74. data = cpu_to_be32(motu->async_handler.offset);
  75. return snd_motu_transaction_write(motu, ASYNC_ADDR_LO, &data,
  76. sizeof(data));
  77. }
  78. int snd_motu_transaction_register(struct snd_motu *motu)
  79. {
  80. static const struct fw_address_region resp_register_region = {
  81. .start = 0xffffe0000000ull,
  82. .end = 0xffffe000ffffull,
  83. };
  84. int err;
  85. /* Perhaps, 4 byte messages are transferred. */
  86. motu->async_handler.length = 4;
  87. motu->async_handler.address_callback = handle_message;
  88. motu->async_handler.callback_data = motu;
  89. err = fw_core_add_address_handler(&motu->async_handler,
  90. &resp_register_region);
  91. if (err < 0)
  92. return err;
  93. err = snd_motu_transaction_reregister(motu);
  94. if (err < 0) {
  95. fw_core_remove_address_handler(&motu->async_handler);
  96. motu->async_handler.address_callback = NULL;
  97. }
  98. return err;
  99. }
  100. void snd_motu_transaction_unregister(struct snd_motu *motu)
  101. {
  102. __be32 data;
  103. if (motu->async_handler.address_callback != NULL)
  104. fw_core_remove_address_handler(&motu->async_handler);
  105. motu->async_handler.address_callback = NULL;
  106. /* Unregister the address. */
  107. data = cpu_to_be32(0x00000000);
  108. snd_motu_transaction_write(motu, ASYNC_ADDR_HI, &data, sizeof(data));
  109. snd_motu_transaction_write(motu, ASYNC_ADDR_LO, &data, sizeof(data));
  110. }