aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/firewire/fw-ohci.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/firewire/fw-ohci.c')
-rw-r--r--drivers/firewire/fw-ohci.c61
1 files changed, 41 insertions, 20 deletions
diff --git a/drivers/firewire/fw-ohci.c b/drivers/firewire/fw-ohci.c
index d6f0644b05d4..8dc872aedce7 100644
--- a/drivers/firewire/fw-ohci.c
+++ b/drivers/firewire/fw-ohci.c
@@ -221,24 +221,48 @@ static void ar_context_tasklet(unsigned long data)
221{ 221{
222 struct ar_context *ctx = (struct ar_context *)data; 222 struct ar_context *ctx = (struct ar_context *)data;
223 struct fw_ohci *ohci = ctx->ohci; 223 struct fw_ohci *ohci = ctx->ohci;
224 u32 status; 224 struct fw_packet p;
225 int length, speed, ack, timestamp, tcode; 225 u32 status, length, tcode;
226 226
227 /* FIXME: What to do about evt_* errors? */ 227 /* FIXME: What to do about evt_* errors? */
228 length = le16_to_cpu(ctx->descriptor.req_count) - 228 length = le16_to_cpu(ctx->descriptor.req_count) -
229 le16_to_cpu(ctx->descriptor.res_count) - 4; 229 le16_to_cpu(ctx->descriptor.res_count) - 4;
230 status = le32_to_cpu(ctx->buffer[length / 4]); 230 status = le32_to_cpu(ctx->buffer[length / 4]);
231 ack = ((status >> 16) & 0x1f) - 16;
232 speed = (status >> 21) & 0x7;
233 timestamp = status & 0xffff;
234 231
235 ctx->buffer[0] = le32_to_cpu(ctx->buffer[0]); 232 p.ack = ((status >> 16) & 0x1f) - 16;
236 ctx->buffer[1] = le32_to_cpu(ctx->buffer[1]); 233 p.speed = (status >> 21) & 0x7;
237 ctx->buffer[2] = le32_to_cpu(ctx->buffer[2]); 234 p.timestamp = status & 0xffff;
235 p.generation = ohci->request_generation;
236
237 p.header[0] = le32_to_cpu(ctx->buffer[0]);
238 p.header[1] = le32_to_cpu(ctx->buffer[1]);
239 p.header[2] = le32_to_cpu(ctx->buffer[2]);
240
241 tcode = (p.header[0] >> 4) & 0x0f;
242 switch (tcode) {
243 case TCODE_WRITE_QUADLET_REQUEST:
244 case TCODE_READ_QUADLET_RESPONSE:
245 p.header[3] = ctx->buffer[3];
246 p.header_length = 16;
247 break;
248
249 case TCODE_WRITE_BLOCK_REQUEST:
250 case TCODE_READ_BLOCK_REQUEST :
251 case TCODE_READ_BLOCK_RESPONSE:
252 case TCODE_LOCK_REQUEST:
253 case TCODE_LOCK_RESPONSE:
254 p.header[3] = le32_to_cpu(ctx->buffer[3]);
255 p.header_length = 16;
256 break;
257
258 case TCODE_WRITE_RESPONSE:
259 case TCODE_READ_QUADLET_REQUEST:
260 p.header_length = 12;
261 break;
262 }
238 263
239 tcode = (ctx->buffer[0] >> 4) & 0x0f; 264 p.payload = (void *) ctx->buffer + p.header_length;
240 if (TCODE_IS_BLOCK_PACKET(tcode)) 265 p.payload_length = length - p.header_length;
241 ctx->buffer[3] = le32_to_cpu(ctx->buffer[3]);
242 266
243 /* The OHCI bus reset handler synthesizes a phy packet with 267 /* The OHCI bus reset handler synthesizes a phy packet with
244 * the new generation number when a bus reset happens (see 268 * the new generation number when a bus reset happens (see
@@ -248,15 +272,12 @@ static void ar_context_tasklet(unsigned long data)
248 * we use the unique tlabel for finding the matching 272 * we use the unique tlabel for finding the matching
249 * request. */ 273 * request. */
250 274
251 if (ack + 16 == 0x09) 275 if (p.ack + 16 == 0x09)
252 ohci->request_generation = (ctx->buffer[2] >> 16) & 0xff; 276 ohci->request_generation = (ctx->buffer[2] >> 16) & 0xff;
253 else if (ctx == &ohci->ar_request_ctx) 277 else if (ctx == &ohci->ar_request_ctx)
254 fw_core_handle_request(&ohci->card, speed, ack, timestamp, 278 fw_core_handle_request(&ohci->card, &p);
255 ohci->request_generation,
256 length, ctx->buffer);
257 else 279 else
258 fw_core_handle_response(&ohci->card, speed, ack, timestamp, 280 fw_core_handle_response(&ohci->card, &p);
259 length, ctx->buffer);
260 281
261 ctx->descriptor.data_address = cpu_to_le32(ctx->buffer_bus); 282 ctx->descriptor.data_address = cpu_to_le32(ctx->buffer_bus);
262 ctx->descriptor.req_count = cpu_to_le16(sizeof ctx->buffer); 283 ctx->descriptor.req_count = cpu_to_le16(sizeof ctx->buffer);
@@ -323,15 +344,15 @@ do_packet_callbacks(struct fw_ohci *ohci, struct list_head *list)
323 struct fw_packet *p, *next; 344 struct fw_packet *p, *next;
324 345
325 list_for_each_entry_safe(p, next, list, link) 346 list_for_each_entry_safe(p, next, list, link)
326 p->callback(p, &ohci->card, p->status); 347 p->callback(p, &ohci->card, p->ack);
327} 348}
328 349
329static void 350static void
330complete_transmission(struct fw_packet *packet, 351complete_transmission(struct fw_packet *packet,
331 int status, struct list_head *list) 352 int ack, struct list_head *list)
332{ 353{
333 list_move_tail(&packet->link, list); 354 list_move_tail(&packet->link, list);
334 packet->status = status; 355 packet->ack = ack;
335} 356}
336 357
337/* This function prepares the first packet in the context queue for 358/* This function prepares the first packet in the context queue for