aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/caif/caif_spi.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/caif/caif_spi.c')
-rw-r--r--drivers/net/caif/caif_spi.c57
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");
33MODULE_AUTHOR("Daniel Martensson<daniel.martensson@stericsson.com>"); 33MODULE_AUTHOR("Daniel Martensson<daniel.martensson@stericsson.com>");
34MODULE_DESCRIPTION("CAIF SPI driver"); 34MODULE_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
36static int spi_loop; 39static int spi_loop;
37module_param(spi_loop, bool, S_IRUGO); 40module_param(spi_loop, bool, S_IRUGO);
38MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode."); 41MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode.");
@@ -41,7 +44,10 @@ MODULE_PARM_DESC(spi_loop, "SPI running in loopback mode.");
41module_param(spi_frm_align, int, S_IRUGO); 44module_param(spi_frm_align, int, S_IRUGO);
42MODULE_PARM_DESC(spi_frm_align, "SPI frame alignment."); 45MODULE_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 */
45module_param(spi_up_head_align, int, S_IRUGO); 51module_param(spi_up_head_align, int, S_IRUGO);
46MODULE_PARM_DESC(spi_up_head_align, "SPI uplink head alignment."); 52MODULE_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
469static void cfspi_xfer_done_cb(struct cfspi_ifc *ifc) 489static 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. */