Commit 5d9cb7d2 authored by Stefan Richter's avatar Stefan Richter

firewire: cdev: add ioctls for iso resource management, amendment

Some fixes:
  - Remove stale documentation.
  - Fix a != vs. == thinko that got in the way of channel management.
  - Try bandwidth deallocation even if channel deallocation failed.

A simplification:
  - fw_cdev_allocate_iso_resource.channels is now ordered like
    libdc1394's dc1394_iso_allocate_channel() channels_allowed
    argument.

By the way, I looked closer at cards from NEC, TI, and VIA, and noticed
that they all don't implement IEEE 1394a behaviour which is meant to
deviate from IEEE 1212's notion of lock compare-swap.  This means that
we have to do two lock transactions instead of one in many cases where
one transaction would already succeed on a fully 1394a compliant IRM.
Signed-off-by: default avatarStefan Richter <stefanr@s5r6.in-berlin.de>
parent 77258da4
...@@ -1082,7 +1082,7 @@ static void iso_resource_work(struct work_struct *work) ...@@ -1082,7 +1082,7 @@ static void iso_resource_work(struct work_struct *work)
spin_unlock_irq(&client->lock); spin_unlock_irq(&client->lock);
if (todo == ISO_RES_ALLOC && channel >= 0) if (todo == ISO_RES_ALLOC && channel >= 0)
r->channels = 1ULL << (63 - channel); r->channels = 1ULL << channel;
if (todo == ISO_RES_REALLOC && success) if (todo == ISO_RES_REALLOC && success)
goto out; goto out;
......
...@@ -204,17 +204,19 @@ static int manage_bandwidth(struct fw_card *card, int irm_id, int generation, ...@@ -204,17 +204,19 @@ static int manage_bandwidth(struct fw_card *card, int irm_id, int generation,
} }
static int manage_channel(struct fw_card *card, int irm_id, int generation, static int manage_channel(struct fw_card *card, int irm_id, int generation,
__be32 channels_mask, u64 offset, bool allocate) u32 channels_mask, u64 offset, bool allocate)
{ {
__be32 data[2], c, old = allocate ? cpu_to_be32(~0) : 0; __be32 data[2], c, all, old;
int i, retry = 5; int i, retry = 5;
old = all = allocate ? cpu_to_be32(~0) : 0;
for (i = 0; i < 32; i++) { for (i = 0; i < 32; i++) {
c = cpu_to_be32(1 << (31 - i)); if (!(channels_mask & 1 << i))
if (!(channels_mask & c))
continue; continue;
if (allocate == !(old & c)) c = cpu_to_be32(1 << (31 - i));
if ((old & c) != (all & c))
continue; continue;
data[0] = old; data[0] = old;
...@@ -233,7 +235,7 @@ static int manage_channel(struct fw_card *card, int irm_id, int generation, ...@@ -233,7 +235,7 @@ static int manage_channel(struct fw_card *card, int irm_id, int generation,
old = data[0]; old = data[0];
/* Is the IRM 1394a-2000 compliant? */ /* Is the IRM 1394a-2000 compliant? */
if ((data[0] & c) != (data[1] & c)) if ((data[0] & c) == (data[1] & c))
continue; continue;
/* 1394-1995 IRM, fall through to retry. */ /* 1394-1995 IRM, fall through to retry. */
...@@ -249,11 +251,10 @@ static int manage_channel(struct fw_card *card, int irm_id, int generation, ...@@ -249,11 +251,10 @@ static int manage_channel(struct fw_card *card, int irm_id, int generation,
static void deallocate_channel(struct fw_card *card, int irm_id, static void deallocate_channel(struct fw_card *card, int irm_id,
int generation, int channel) int generation, int channel)
{ {
__be32 mask; u32 mask;
u64 offset; u64 offset;
mask = channel < 32 ? cpu_to_be32(1 << (31 - channel)) : mask = channel < 32 ? 1 << channel : 1 << (channel - 32);
cpu_to_be32(1 << (63 - channel));
offset = channel < 32 ? CSR_REGISTER_BASE + CSR_CHANNELS_AVAILABLE_HI : offset = channel < 32 ? CSR_REGISTER_BASE + CSR_CHANNELS_AVAILABLE_HI :
CSR_REGISTER_BASE + CSR_CHANNELS_AVAILABLE_LO; CSR_REGISTER_BASE + CSR_CHANNELS_AVAILABLE_LO;
...@@ -266,7 +267,12 @@ static void deallocate_channel(struct fw_card *card, int irm_id, ...@@ -266,7 +267,12 @@ static void deallocate_channel(struct fw_card *card, int irm_id,
* In parameters: card, generation, channels_mask, bandwidth, allocate * In parameters: card, generation, channels_mask, bandwidth, allocate
* Out parameters: channel, bandwidth * Out parameters: channel, bandwidth
* This function blocks (sleeps) during communication with the IRM. * This function blocks (sleeps) during communication with the IRM.
*
* Allocates or deallocates at most one channel out of channels_mask. * Allocates or deallocates at most one channel out of channels_mask.
* channels_mask is a bitfield with MSB for channel 63 and LSB for channel 0.
* (Note, the IRM's CHANNELS_AVAILABLE is a big-endian bitfield with MSB for
* channel 0 and LSB for channel 63.)
* Allocates or deallocates as many bandwidth allocation units as specified.
* *
* Returns channel < 0 if no channel was allocated or deallocated. * Returns channel < 0 if no channel was allocated or deallocated.
* Returns bandwidth = 0 if no bandwidth was allocated or deallocated. * Returns bandwidth = 0 if no bandwidth was allocated or deallocated.
...@@ -274,17 +280,17 @@ static void deallocate_channel(struct fw_card *card, int irm_id, ...@@ -274,17 +280,17 @@ static void deallocate_channel(struct fw_card *card, int irm_id,
* If generation is stale, deallocations succeed but allocations fail with * If generation is stale, deallocations succeed but allocations fail with
* channel = -EAGAIN. * channel = -EAGAIN.
* *
* If channel (de)allocation fails, bandwidth (de)allocation fails too. * If channel allocation fails, no bandwidth will be allocated either.
* If bandwidth allocation fails, no channel will be allocated either. * If bandwidth allocation fails, no channel will be allocated either.
* If bandwidth deallocation fails, channel deallocation may still have been * But deallocations of channel and bandwidth are tried independently
* successful. * of each other's success.
*/ */
void fw_iso_resource_manage(struct fw_card *card, int generation, void fw_iso_resource_manage(struct fw_card *card, int generation,
u64 channels_mask, int *channel, int *bandwidth, u64 channels_mask, int *channel, int *bandwidth,
bool allocate) bool allocate)
{ {
__be32 channels_hi = cpu_to_be32(channels_mask >> 32); u32 channels_hi = channels_mask; /* channels 31...0 */
__be32 channels_lo = cpu_to_be32(channels_mask); u32 channels_lo = channels_mask >> 32; /* channels 63...32 */
int irm_id, ret, c = -EINVAL; int irm_id, ret, c = -EINVAL;
spin_lock_irq(&card->lock); spin_lock_irq(&card->lock);
...@@ -302,7 +308,7 @@ void fw_iso_resource_manage(struct fw_card *card, int generation, ...@@ -302,7 +308,7 @@ void fw_iso_resource_manage(struct fw_card *card, int generation,
} }
*channel = c; *channel = c;
if (channels_mask != 0 && c < 0) if (allocate && channels_mask != 0 && c < 0)
*bandwidth = 0; *bandwidth = 0;
if (*bandwidth == 0) if (*bandwidth == 0)
...@@ -312,7 +318,7 @@ void fw_iso_resource_manage(struct fw_card *card, int generation, ...@@ -312,7 +318,7 @@ void fw_iso_resource_manage(struct fw_card *card, int generation,
if (ret < 0) if (ret < 0)
*bandwidth = 0; *bandwidth = 0;
if (ret < 0 && c >= 0 && allocate) { if (allocate && ret < 0 && c >= 0) {
deallocate_channel(card, irm_id, generation, c); deallocate_channel(card, irm_id, generation, c);
*channel = ret; *channel = ret;
} }
......
...@@ -174,8 +174,6 @@ struct fw_cdev_event_iso_interrupt { ...@@ -174,8 +174,6 @@ struct fw_cdev_event_iso_interrupt {
* @handle: Reference by which an allocated resource can be deallocated * @handle: Reference by which an allocated resource can be deallocated
* @channel: Isochronous channel which was (de)allocated, if any * @channel: Isochronous channel which was (de)allocated, if any
* @bandwidth: Bandwidth allocation units which were (de)allocated, if any * @bandwidth: Bandwidth allocation units which were (de)allocated, if any
* @channels_available: Last known availability of channels
* @bandwidth_available: Last known availability of bandwidth
* *
* An %FW_CDEV_EVENT_ISO_RESOURCE_ALLOCATED event is sent after an isochronous * An %FW_CDEV_EVENT_ISO_RESOURCE_ALLOCATED event is sent after an isochronous
* resource was allocated at the IRM. The client has to check @channel and * resource was allocated at the IRM. The client has to check @channel and
...@@ -580,7 +578,7 @@ struct fw_cdev_get_cycle_timer { ...@@ -580,7 +578,7 @@ struct fw_cdev_get_cycle_timer {
* *
* The %FW_CDEV_IOC_DEALLOCATE_ISO_RESOURCE_ONCE ioctl works like * The %FW_CDEV_IOC_DEALLOCATE_ISO_RESOURCE_ONCE ioctl works like
* %FW_CDEV_IOC_ALLOCATE_ISO_RESOURCE_ONCE except that resources are freed * %FW_CDEV_IOC_ALLOCATE_ISO_RESOURCE_ONCE except that resources are freed
* instead of allocated. At most one channel may be specified in this ioctl. * instead of allocated.
* An %FW_CDEV_EVENT_ISO_RESOURCE_DEALLOCATED event concludes this operation. * An %FW_CDEV_EVENT_ISO_RESOURCE_DEALLOCATED event concludes this operation.
* *
* To summarize, %FW_CDEV_IOC_DEALLOCATE_ISO_RESOURCE allocates iso resources * To summarize, %FW_CDEV_IOC_DEALLOCATE_ISO_RESOURCE allocates iso resources
...@@ -588,9 +586,9 @@ struct fw_cdev_get_cycle_timer { ...@@ -588,9 +586,9 @@ struct fw_cdev_get_cycle_timer {
* In contrast, %FW_CDEV_IOC_ALLOCATE_ISO_RESOURCE_ONCE allocates iso resources * In contrast, %FW_CDEV_IOC_ALLOCATE_ISO_RESOURCE_ONCE allocates iso resources
* for the duration of a bus generation. * for the duration of a bus generation.
* *
* @channels is a host-endian bitfield with the most significant bit * @channels is a host-endian bitfield with the least significant bit
* representing channel 0 and the least significant bit representing channel 63: * representing channel 0 and the most significant bit representing channel 63:
* 1ULL << (63 - c) * 1ULL << c for each channel c that is a candidate for (de)allocation.
* *
* @bandwidth is expressed in bandwidth allocation units, i.e. the time to send * @bandwidth is expressed in bandwidth allocation units, i.e. the time to send
* one quadlet of data (payload or header data) at speed S1600. * one quadlet of data (payload or header data) at speed S1600.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment