Blame view

sound/firewire/lib.c 3.1 KB
da607e196   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-only
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
2
3
4
5
  /*
   * miscellaneous helper functions
   *
   * Copyright (c) Clemens Ladisch <clemens@ladisch.de>
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
6
7
8
9
10
11
   */
  
  #include <linux/delay.h>
  #include <linux/device.h>
  #include <linux/firewire.h>
  #include <linux/module.h>
585d7cba5   Takashi Sakamoto   ALSA: firewire-li...
12
  #include <linux/slab.h>
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
13
  #include "lib.h"
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
14
  #define ERROR_RETRY_DELAY_MS	20
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
15
16
  
  /**
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
17
18
19
20
21
22
   * snd_fw_transaction - send a request and wait for its completion
   * @unit: the driver's unit on the target device
   * @tcode: the transaction code
   * @offset: the address in the target's address space
   * @buffer: input/output data
   * @length: length of @buffer
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
23
24
25
   * @flags: use %FW_FIXED_GENERATION and add the generation value to attempt the
   *         request only in that generation; use %FW_QUIET to suppress error
   *         messages
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
26
27
28
29
30
31
32
   *
   * Submits an asynchronous request to the target device, and waits for the
   * response.  The node ID and the current generation are derived from @unit.
   * On a bus reset or an error, the transaction is retried a few times.
   * Returns zero on success, or a negative error code.
   */
  int snd_fw_transaction(struct fw_unit *unit, int tcode,
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
33
34
  		       u64 offset, void *buffer, size_t length,
  		       unsigned int flags)
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
35
36
37
  {
  	struct fw_device *device = fw_parent_device(unit);
  	int generation, rcode, tries = 0;
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
38
  	generation = flags & FW_GENERATION_MASK;
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
39
  	for (;;) {
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
40
41
42
43
  		if (!(flags & FW_FIXED_GENERATION)) {
  			generation = device->generation;
  			smp_rmb(); /* node_id vs. generation */
  		}
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
44
45
46
47
48
49
50
  		rcode = fw_run_transaction(device->card, tcode,
  					   device->node_id, generation,
  					   device->max_speed, offset,
  					   buffer, length);
  
  		if (rcode == RCODE_COMPLETE)
  			return 0;
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
51
52
  		if (rcode == RCODE_GENERATION && (flags & FW_FIXED_GENERATION))
  			return -EAGAIN;
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
53
  		if (rcode_is_permanent_error(rcode) || ++tries >= 3) {
1b70485f1   Clemens Ladisch   ALSA: firewire: e...
54
55
56
57
58
  			if (!(flags & FW_QUIET))
  				dev_err(&unit->device,
  					"transaction failed: %s
  ",
  					fw_rcode_string(rcode));
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
59
60
61
62
63
64
65
  			return -EIO;
  		}
  
  		msleep(ERROR_RETRY_DELAY_MS);
  	}
  }
  EXPORT_SYMBOL(snd_fw_transaction);
923f92ebb   Takashi Sakamoto   ALSA: firewire-li...
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
  #define PROBE_DELAY_MS		(2 * MSEC_PER_SEC)
  
  /**
   * snd_fw_schedule_registration - schedule work for sound card registration
   * @unit: an instance for unit on IEEE 1394 bus
   * @dwork: delayed work with callback function
   *
   * This function is not designed for general purposes. When new unit is
   * connected to IEEE 1394 bus, the bus is under bus-reset state because of
   * topological change. In this state, units tend to fail both of asynchronous
   * and isochronous communication. To avoid this problem, this function is used
   * to postpone sound card registration after the state. The callers must
   * set up instance of delayed work in advance.
   */
  void snd_fw_schedule_registration(struct fw_unit *unit,
  				  struct delayed_work *dwork)
  {
  	u64 now, delay;
  
  	now = get_jiffies_64();
  	delay = fw_parent_device(unit)->card->reset_jiffies
  					+ msecs_to_jiffies(PROBE_DELAY_MS);
  
  	if (time_after64(delay, now))
  		delay -= now;
  	else
  		delay = 0;
  
  	mod_delayed_work(system_wq, dwork, delay);
  }
  EXPORT_SYMBOL(snd_fw_schedule_registration);
31ef9134e   Clemens Ladisch   ALSA: add LaCie F...
97
98
99
  MODULE_DESCRIPTION("FireWire audio helper functions");
  MODULE_AUTHOR("Clemens Ladisch <clemens@ladisch.de>");
  MODULE_LICENSE("GPL v2");