diff options
Diffstat (limited to 'drivers/firewire')
-rw-r--r-- | drivers/firewire/fw-ohci.c | 99 |
1 files changed, 49 insertions, 50 deletions
diff --git a/drivers/firewire/fw-ohci.c b/drivers/firewire/fw-ohci.c index a44d16d0c505..a9f2d07e7c65 100644 --- a/drivers/firewire/fw-ohci.c +++ b/drivers/firewire/fw-ohci.c | |||
@@ -1461,24 +1461,24 @@ static int handle_ir_packet_per_buffer(struct context *context, | |||
1461 | { | 1461 | { |
1462 | struct iso_context *ctx = | 1462 | struct iso_context *ctx = |
1463 | container_of(context, struct iso_context, context); | 1463 | container_of(context, struct iso_context, context); |
1464 | struct descriptor *pd = d + 1; | 1464 | struct descriptor *pd; |
1465 | __le32 *ir_header; | 1465 | __le32 *ir_header; |
1466 | size_t header_length; | 1466 | void *p; |
1467 | void *p, *end; | 1467 | int i; |
1468 | int i, z; | ||
1469 | 1468 | ||
1470 | if (pd->res_count == pd->req_count) | 1469 | for (pd = d; pd <= last; pd++) { |
1470 | if (pd->transfer_status) | ||
1471 | break; | ||
1472 | } | ||
1473 | if (pd > last) | ||
1471 | /* Descriptor(s) not done yet, stop iteration */ | 1474 | /* Descriptor(s) not done yet, stop iteration */ |
1472 | return 0; | 1475 | return 0; |
1473 | 1476 | ||
1474 | header_length = le16_to_cpu(d->req_count); | ||
1475 | |||
1476 | i = ctx->header_length; | 1477 | i = ctx->header_length; |
1477 | z = le32_to_cpu(pd->branch_address) & 0xf; | 1478 | p = last + 1; |
1478 | p = d + z; | ||
1479 | end = p + header_length; | ||
1480 | 1479 | ||
1481 | while (p < end && i + ctx->base.header_size <= PAGE_SIZE) { | 1480 | if (ctx->base.header_size > 0 && |
1481 | i + ctx->base.header_size <= PAGE_SIZE) { | ||
1482 | /* | 1482 | /* |
1483 | * The iso header is byteswapped to little endian by | 1483 | * The iso header is byteswapped to little endian by |
1484 | * the controller, but the remaining header quadlets | 1484 | * the controller, but the remaining header quadlets |
@@ -1487,14 +1487,11 @@ static int handle_ir_packet_per_buffer(struct context *context, | |||
1487 | */ | 1487 | */ |
1488 | *(u32 *) (ctx->header + i) = __swab32(*(u32 *) (p + 4)); | 1488 | *(u32 *) (ctx->header + i) = __swab32(*(u32 *) (p + 4)); |
1489 | memcpy(ctx->header + i + 4, p + 8, ctx->base.header_size - 4); | 1489 | memcpy(ctx->header + i + 4, p + 8, ctx->base.header_size - 4); |
1490 | i += ctx->base.header_size; | 1490 | ctx->header_length += ctx->base.header_size; |
1491 | p += ctx->base.header_size + 4; | ||
1492 | } | 1491 | } |
1493 | 1492 | ||
1494 | ctx->header_length = i; | 1493 | if (le16_to_cpu(last->control) & DESCRIPTOR_IRQ_ALWAYS) { |
1495 | 1494 | ir_header = (__le32 *) p; | |
1496 | if (le16_to_cpu(pd->control) & DESCRIPTOR_IRQ_ALWAYS) { | ||
1497 | ir_header = (__le32 *) (d + z); | ||
1498 | ctx->base.callback(&ctx->base, | 1495 | ctx->base.callback(&ctx->base, |
1499 | le32_to_cpu(ir_header[0]) & 0xffff, | 1496 | le32_to_cpu(ir_header[0]) & 0xffff, |
1500 | ctx->header_length, ctx->header, | 1497 | ctx->header_length, ctx->header, |
@@ -1502,7 +1499,6 @@ static int handle_ir_packet_per_buffer(struct context *context, | |||
1502 | ctx->header_length = 0; | 1499 | ctx->header_length = 0; |
1503 | } | 1500 | } |
1504 | 1501 | ||
1505 | |||
1506 | return 1; | 1502 | return 1; |
1507 | } | 1503 | } |
1508 | 1504 | ||
@@ -1853,67 +1849,70 @@ ohci_queue_iso_receive_packet_per_buffer(struct fw_iso_context *base, | |||
1853 | { | 1849 | { |
1854 | struct iso_context *ctx = container_of(base, struct iso_context, base); | 1850 | struct iso_context *ctx = container_of(base, struct iso_context, base); |
1855 | struct descriptor *d = NULL, *pd = NULL; | 1851 | struct descriptor *d = NULL, *pd = NULL; |
1856 | struct fw_iso_packet *p; | 1852 | struct fw_iso_packet *p = packet; |
1857 | dma_addr_t d_bus, page_bus; | 1853 | dma_addr_t d_bus, page_bus; |
1858 | u32 z, header_z, rest; | 1854 | u32 z, header_z, rest; |
1859 | int i, page, offset, packet_count, header_size; | 1855 | int i, j, length; |
1860 | 1856 | int page, offset, packet_count, header_size, payload_per_buffer; | |
1861 | if (packet->skip) { | ||
1862 | d = context_get_descriptors(&ctx->context, 1, &d_bus); | ||
1863 | if (d == NULL) | ||
1864 | return -ENOMEM; | ||
1865 | |||
1866 | d->control = cpu_to_le16(DESCRIPTOR_STATUS | | ||
1867 | DESCRIPTOR_INPUT_LAST | | ||
1868 | DESCRIPTOR_BRANCH_ALWAYS | | ||
1869 | DESCRIPTOR_WAIT); | ||
1870 | context_append(&ctx->context, d, 1, 0); | ||
1871 | } | ||
1872 | |||
1873 | /* one descriptor for header, one for payload */ | ||
1874 | /* FIXME: handle cases where we need multiple desc. for payload */ | ||
1875 | z = 2; | ||
1876 | p = packet; | ||
1877 | 1857 | ||
1878 | /* | 1858 | /* |
1879 | * The OHCI controller puts the status word in the | 1859 | * The OHCI controller puts the status word in the |
1880 | * buffer too, so we need 4 extra bytes per packet. | 1860 | * buffer too, so we need 4 extra bytes per packet. |
1881 | */ | 1861 | */ |
1882 | packet_count = p->header_length / ctx->base.header_size; | 1862 | packet_count = p->header_length / ctx->base.header_size; |
1883 | header_size = packet_count * (ctx->base.header_size + 4); | 1863 | header_size = ctx->base.header_size + 4; |
1884 | 1864 | ||
1885 | /* Get header size in number of descriptors. */ | 1865 | /* Get header size in number of descriptors. */ |
1886 | header_z = DIV_ROUND_UP(header_size, sizeof(*d)); | 1866 | header_z = DIV_ROUND_UP(header_size, sizeof(*d)); |
1887 | page = payload >> PAGE_SHIFT; | 1867 | page = payload >> PAGE_SHIFT; |
1888 | offset = payload & ~PAGE_MASK; | 1868 | offset = payload & ~PAGE_MASK; |
1889 | rest = p->payload_length; | 1869 | payload_per_buffer = p->payload_length / packet_count; |
1890 | 1870 | ||
1891 | for (i = 0; i < packet_count; i++) { | 1871 | for (i = 0; i < packet_count; i++) { |
1892 | /* d points to the header descriptor */ | 1872 | /* d points to the header descriptor */ |
1873 | z = DIV_ROUND_UP(payload_per_buffer + offset, PAGE_SIZE) + 1; | ||
1893 | d = context_get_descriptors(&ctx->context, | 1874 | d = context_get_descriptors(&ctx->context, |
1894 | z + header_z, &d_bus); | 1875 | z + header_z, &d_bus); |
1895 | if (d == NULL) | 1876 | if (d == NULL) |
1896 | return -ENOMEM; | 1877 | return -ENOMEM; |
1897 | 1878 | ||
1898 | d->control = cpu_to_le16(DESCRIPTOR_INPUT_MORE); | 1879 | d->control = cpu_to_le16(DESCRIPTOR_STATUS | |
1880 | DESCRIPTOR_INPUT_MORE); | ||
1881 | if (p->skip && i == 0) | ||
1882 | d->control |= cpu_to_le16(DESCRIPTOR_WAIT); | ||
1899 | d->req_count = cpu_to_le16(header_size); | 1883 | d->req_count = cpu_to_le16(header_size); |
1900 | d->res_count = d->req_count; | 1884 | d->res_count = d->req_count; |
1885 | d->transfer_status = 0; | ||
1901 | d->data_address = cpu_to_le32(d_bus + (z * sizeof(*d))); | 1886 | d->data_address = cpu_to_le32(d_bus + (z * sizeof(*d))); |
1902 | 1887 | ||
1903 | /* pd points to the payload descriptor */ | 1888 | rest = payload_per_buffer; |
1904 | pd = d + 1; | 1889 | for (j = 1; j < z; j++) { |
1890 | pd = d + j; | ||
1891 | pd->control = cpu_to_le16(DESCRIPTOR_STATUS | | ||
1892 | DESCRIPTOR_INPUT_MORE); | ||
1893 | |||
1894 | if (offset + rest < PAGE_SIZE) | ||
1895 | length = rest; | ||
1896 | else | ||
1897 | length = PAGE_SIZE - offset; | ||
1898 | pd->req_count = cpu_to_le16(length); | ||
1899 | pd->res_count = pd->req_count; | ||
1900 | pd->transfer_status = 0; | ||
1901 | |||
1902 | page_bus = page_private(buffer->pages[page]); | ||
1903 | pd->data_address = cpu_to_le32(page_bus + offset); | ||
1904 | |||
1905 | offset = (offset + length) & ~PAGE_MASK; | ||
1906 | rest -= length; | ||
1907 | if (offset == 0) | ||
1908 | page++; | ||
1909 | } | ||
1905 | pd->control = cpu_to_le16(DESCRIPTOR_STATUS | | 1910 | pd->control = cpu_to_le16(DESCRIPTOR_STATUS | |
1906 | DESCRIPTOR_INPUT_LAST | | 1911 | DESCRIPTOR_INPUT_LAST | |
1907 | DESCRIPTOR_BRANCH_ALWAYS); | 1912 | DESCRIPTOR_BRANCH_ALWAYS); |
1908 | if (p->interrupt) | 1913 | if (p->interrupt && i == packet_count - 1) |
1909 | pd->control |= cpu_to_le16(DESCRIPTOR_IRQ_ALWAYS); | 1914 | pd->control |= cpu_to_le16(DESCRIPTOR_IRQ_ALWAYS); |
1910 | 1915 | ||
1911 | pd->req_count = cpu_to_le16(rest); | ||
1912 | pd->res_count = pd->req_count; | ||
1913 | |||
1914 | page_bus = page_private(buffer->pages[page]); | ||
1915 | pd->data_address = cpu_to_le32(page_bus + offset); | ||
1916 | |||
1917 | context_append(&ctx->context, d, z, header_z); | 1916 | context_append(&ctx->context, d, z, header_z); |
1918 | } | 1917 | } |
1919 | 1918 | ||