diff options
Diffstat (limited to 'drivers/net/caif/caif_spi.c')
-rw-r--r-- | drivers/net/caif/caif_spi.c | 57 |
1 files changed, 43 insertions, 14 deletions
diff --git a/drivers/net/caif/caif_spi.c b/drivers/net/caif/caif_spi.c index f5058ff2b210..57e639373815 100644 --- a/drivers/net/caif/caif_spi.c +++ b/drivers/net/caif/caif_spi.c | |||
@@ -33,6 +33,9 @@ MODULE_LICENSE("GPL"); | |||
33 | MODULE_AUTHOR("Daniel Martensson<daniel.martensson@stericsson.com>"); | 33 | MODULE_AUTHOR("Daniel Martensson<daniel.martensson@stericsson.com>"); |
34 | MODULE_DESCRIPTION("CAIF SPI driver"); | 34 | MODULE_DESCRIPTION("CAIF SPI driver"); |
35 | 35 | ||
36 | /* Returns the number of padding bytes for alignment. */ | ||
37 | #define PAD_POW2(x, pow) ((((x)&((pow)-1))==0) ? 0 : (((pow)-((x)&((pow)-1))))) | ||
38 | |||
36 | static int spi_loop; | 39 | static int spi_loop; |
37 | module_param(spi_loop, bool, S_IRUGO); | 40 | module_param(spi_loop, bool, S_IRUGO); |
38 | MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode."); | 41 | MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode."); |
@@ -41,7 +44,10 @@ MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode."); | |||
41 | module_param(spi_frm_align, int, S_IRUGO); | 44 | module_param(spi_frm_align, int, S_IRUGO); |
42 | MODULE_PARM_DESC(spi_frm_align, "SPI frame alignment."); | 45 | MODULE_PARM_DESC(spi_frm_align, "SPI frame alignment."); |
43 | 46 | ||
44 | /* SPI padding options. */ | 47 | /* |
48 | * SPI padding options. | ||
49 | * Warning: must be a base of 2 (& operation used) and can not be zero ! | ||
50 | */ | ||
45 | module_param(spi_up_head_align, int, S_IRUGO); | 51 | module_param(spi_up_head_align, int, S_IRUGO); |
46 | MODULE_PARM_DESC(spi_up_head_align, "SPI uplink head alignment."); | 52 | MODULE_PARM_DESC(spi_up_head_align, "SPI uplink head alignment."); |
47 | 53 | ||
@@ -335,6 +341,9 @@ int cfspi_xmitfrm(struct cfspi *cfspi, u8 *buf, size_t len) | |||
335 | u8 *dst = buf; | 341 | u8 *dst = buf; |
336 | caif_assert(buf); | 342 | caif_assert(buf); |
337 | 343 | ||
344 | if (cfspi->slave && !cfspi->slave_talked) | ||
345 | cfspi->slave_talked = true; | ||
346 | |||
338 | do { | 347 | do { |
339 | struct sk_buff *skb; | 348 | struct sk_buff *skb; |
340 | struct caif_payload_info *info; | 349 | struct caif_payload_info *info; |
@@ -355,8 +364,8 @@ int cfspi_xmitfrm(struct cfspi *cfspi, u8 *buf, size_t len) | |||
355 | * Compute head offset i.e. number of bytes to add to | 364 | * Compute head offset i.e. number of bytes to add to |
356 | * get the start of the payload aligned. | 365 | * get the start of the payload aligned. |
357 | */ | 366 | */ |
358 | if (spi_up_head_align) { | 367 | if (spi_up_head_align > 1) { |
359 | spad = 1 + ((info->hdr_len + 1) & spi_up_head_align); | 368 | spad = 1 + PAD_POW2((info->hdr_len + 1), spi_up_head_align); |
360 | *dst = (u8)(spad - 1); | 369 | *dst = (u8)(spad - 1); |
361 | dst += spad; | 370 | dst += spad; |
362 | } | 371 | } |
@@ -371,7 +380,7 @@ int cfspi_xmitfrm(struct cfspi *cfspi, u8 *buf, size_t len) | |||
371 | * Compute tail offset i.e. number of bytes to add to | 380 | * Compute tail offset i.e. number of bytes to add to |
372 | * get the complete CAIF frame aligned. | 381 | * get the complete CAIF frame aligned. |
373 | */ | 382 | */ |
374 | epad = (skb->len + spad) & spi_up_tail_align; | 383 | epad = PAD_POW2((skb->len + spad), spi_up_tail_align); |
375 | dst += epad; | 384 | dst += epad; |
376 | 385 | ||
377 | dev_kfree_skb(skb); | 386 | dev_kfree_skb(skb); |
@@ -388,7 +397,7 @@ int cfspi_xmitlen(struct cfspi *cfspi) | |||
388 | int pkts = 0; | 397 | int pkts = 0; |
389 | 398 | ||
390 | /* | 399 | /* |
391 | * Decommit previously commited frames. | 400 | * Decommit previously committed frames. |
392 | * skb_queue_splice_tail(&cfspi->chead,&cfspi->qhead) | 401 | * skb_queue_splice_tail(&cfspi->chead,&cfspi->qhead) |
393 | */ | 402 | */ |
394 | while (skb_peek(&cfspi->chead)) { | 403 | while (skb_peek(&cfspi->chead)) { |
@@ -415,14 +424,14 @@ int cfspi_xmitlen(struct cfspi *cfspi) | |||
415 | * Compute head offset i.e. number of bytes to add to | 424 | * Compute head offset i.e. number of bytes to add to |
416 | * get the start of the payload aligned. | 425 | * get the start of the payload aligned. |
417 | */ | 426 | */ |
418 | if (spi_up_head_align) | 427 | if (spi_up_head_align > 1) |
419 | spad = 1 + ((info->hdr_len + 1) & spi_up_head_align); | 428 | spad = 1 + PAD_POW2((info->hdr_len + 1), spi_up_head_align); |
420 | 429 | ||
421 | /* | 430 | /* |
422 | * Compute tail offset i.e. number of bytes to add to | 431 | * Compute tail offset i.e. number of bytes to add to |
423 | * get the complete CAIF frame aligned. | 432 | * get the complete CAIF frame aligned. |
424 | */ | 433 | */ |
425 | epad = (skb->len + spad) & spi_up_tail_align; | 434 | epad = PAD_POW2((skb->len + spad), spi_up_tail_align); |
426 | 435 | ||
427 | if ((skb->len + spad + epad + frm_len) <= CAIF_MAX_SPI_FRAME) { | 436 | if ((skb->len + spad + epad + frm_len) <= CAIF_MAX_SPI_FRAME) { |
428 | skb_queue_tail(&cfspi->chead, skb); | 437 | skb_queue_tail(&cfspi->chead, skb); |
@@ -431,6 +440,7 @@ int cfspi_xmitlen(struct cfspi *cfspi) | |||
431 | } else { | 440 | } else { |
432 | /* Put back packet. */ | 441 | /* Put back packet. */ |
433 | skb_queue_head(&cfspi->qhead, skb); | 442 | skb_queue_head(&cfspi->qhead, skb); |
443 | break; | ||
434 | } | 444 | } |
435 | } while (pkts <= CAIF_MAX_SPI_PKTS); | 445 | } while (pkts <= CAIF_MAX_SPI_PKTS); |
436 | 446 | ||
@@ -451,6 +461,15 @@ static void cfspi_ss_cb(bool assert, struct cfspi_ifc *ifc) | |||
451 | { | 461 | { |
452 | struct cfspi *cfspi = (struct cfspi *)ifc->priv; | 462 | struct cfspi *cfspi = (struct cfspi *)ifc->priv; |
453 | 463 | ||
464 | /* | ||
465 | * The slave device is the master on the link. Interrupts before the | ||
466 | * slave has transmitted are considered spurious. | ||
467 | */ | ||
468 | if (cfspi->slave && !cfspi->slave_talked) { | ||
469 | printk(KERN_WARNING "CFSPI: Spurious SS interrupt.\n"); | ||
470 | return; | ||
471 | } | ||
472 | |||
454 | if (!in_interrupt()) | 473 | if (!in_interrupt()) |
455 | spin_lock(&cfspi->lock); | 474 | spin_lock(&cfspi->lock); |
456 | if (assert) { | 475 | if (assert) { |
@@ -463,7 +482,8 @@ static void cfspi_ss_cb(bool assert, struct cfspi_ifc *ifc) | |||
463 | spin_unlock(&cfspi->lock); | 482 | spin_unlock(&cfspi->lock); |
464 | 483 | ||
465 | /* Wake up the xfer thread. */ | 484 | /* Wake up the xfer thread. */ |
466 | wake_up_interruptible(&cfspi->wait); | 485 | if (assert) |
486 | wake_up_interruptible(&cfspi->wait); | ||
467 | } | 487 | } |
468 | 488 | ||
469 | static void cfspi_xfer_done_cb(struct cfspi_ifc *ifc) | 489 | static void cfspi_xfer_done_cb(struct cfspi_ifc *ifc) |
@@ -521,7 +541,7 @@ int cfspi_rxfrm(struct cfspi *cfspi, u8 *buf, size_t len) | |||
521 | * Compute head offset i.e. number of bytes added to | 541 | * Compute head offset i.e. number of bytes added to |
522 | * get the start of the payload aligned. | 542 | * get the start of the payload aligned. |
523 | */ | 543 | */ |
524 | if (spi_down_head_align) { | 544 | if (spi_down_head_align > 1) { |
525 | spad = 1 + *src; | 545 | spad = 1 + *src; |
526 | src += spad; | 546 | src += spad; |
527 | } | 547 | } |
@@ -562,7 +582,7 @@ int cfspi_rxfrm(struct cfspi *cfspi, u8 *buf, size_t len) | |||
562 | * Compute tail offset i.e. number of bytes added to | 582 | * Compute tail offset i.e. number of bytes added to |
563 | * get the complete CAIF frame aligned. | 583 | * get the complete CAIF frame aligned. |
564 | */ | 584 | */ |
565 | epad = (pkt_len + spad) & spi_down_tail_align; | 585 | epad = PAD_POW2((pkt_len + spad), spi_down_tail_align); |
566 | src += epad; | 586 | src += epad; |
567 | } while ((src - buf) < len); | 587 | } while ((src - buf) < len); |
568 | 588 | ||
@@ -615,19 +635,28 @@ int cfspi_spi_probe(struct platform_device *pdev) | |||
615 | 635 | ||
616 | ndev = alloc_netdev(sizeof(struct cfspi), | 636 | ndev = alloc_netdev(sizeof(struct cfspi), |
617 | "cfspi%d", cfspi_setup); | 637 | "cfspi%d", cfspi_setup); |
618 | if (!dev) | 638 | if (!ndev) |
619 | return -ENODEV; | 639 | return -ENOMEM; |
620 | 640 | ||
621 | cfspi = netdev_priv(ndev); | 641 | cfspi = netdev_priv(ndev); |
622 | netif_stop_queue(ndev); | 642 | netif_stop_queue(ndev); |
623 | cfspi->ndev = ndev; | 643 | cfspi->ndev = ndev; |
624 | cfspi->pdev = pdev; | 644 | cfspi->pdev = pdev; |
625 | 645 | ||
626 | /* Set flow info */ | 646 | /* Set flow info. */ |
627 | cfspi->flow_off_sent = 0; | 647 | cfspi->flow_off_sent = 0; |
628 | cfspi->qd_low_mark = LOW_WATER_MARK; | 648 | cfspi->qd_low_mark = LOW_WATER_MARK; |
629 | cfspi->qd_high_mark = HIGH_WATER_MARK; | 649 | cfspi->qd_high_mark = HIGH_WATER_MARK; |
630 | 650 | ||
651 | /* Set slave info. */ | ||
652 | if (!strncmp(cfspi_spi_driver.driver.name, "cfspi_sspi", 10)) { | ||
653 | cfspi->slave = true; | ||
654 | cfspi->slave_talked = false; | ||
655 | } else { | ||
656 | cfspi->slave = false; | ||
657 | cfspi->slave_talked = false; | ||
658 | } | ||
659 | |||
631 | /* Assign the SPI device. */ | 660 | /* Assign the SPI device. */ |
632 | cfspi->dev = dev; | 661 | cfspi->dev = dev; |
633 | /* Assign the device ifc to this SPI interface. */ | 662 | /* Assign the device ifc to this SPI interface. */ |