diff options
author | Kim Phillips <kim.phillips@freescale.com> | 2011-11-21 03:13:27 -0500 |
---|---|---|
committer | Herbert Xu <herbert@gondor.apana.org.au> | 2011-11-21 03:21:51 -0500 |
commit | c3e337f88a5b3784cb3c806ffd650d06adff1ea5 (patch) | |
tree | 12ca78132c4ccc74257bf2ad634e1c06a062d2bd /drivers/crypto/talitos.c | |
parent | ad42d5fc85383278663ecb58a24f6547ad0ba735 (diff) |
crypto: talitos - support for channel remap and 2nd IRQ
Some later SEC v3.x are equipped with a second IRQ line.
By correctly assigning IRQ affinity, this feature can be
used to increase performance on dual core parts, like the
MPC8572E and P2020E.
The existence of the 2nd IRQ is determined from the device
node's interrupt property. If present, the driver remaps
two of four channels, which in turn makes those channels
trigger their interrupts on the 2nd line instead of the first.
To handle single- and dual-IRQ combinations efficiently,
talitos gets two new interrupt handlers and back-half workers.
[includes a fix to MCR_LO's address.]
Signed-off-by: Kim Phillips <kim.phillips@freescale.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
Diffstat (limited to 'drivers/crypto/talitos.c')
-rw-r--r-- | drivers/crypto/talitos.c | 207 |
1 files changed, 138 insertions, 69 deletions
diff --git a/drivers/crypto/talitos.c b/drivers/crypto/talitos.c index 7f82e91e461c..92c0ca75400d 100644 --- a/drivers/crypto/talitos.c +++ b/drivers/crypto/talitos.c | |||
@@ -122,7 +122,7 @@ struct talitos_private { | |||
122 | struct device *dev; | 122 | struct device *dev; |
123 | struct platform_device *ofdev; | 123 | struct platform_device *ofdev; |
124 | void __iomem *reg; | 124 | void __iomem *reg; |
125 | int irq; | 125 | int irq[2]; |
126 | 126 | ||
127 | /* SEC version geometry (from device tree node) */ | 127 | /* SEC version geometry (from device tree node) */ |
128 | unsigned int num_channels; | 128 | unsigned int num_channels; |
@@ -146,7 +146,7 @@ struct talitos_private { | |||
146 | atomic_t last_chan ____cacheline_aligned; | 146 | atomic_t last_chan ____cacheline_aligned; |
147 | 147 | ||
148 | /* request callback tasklet */ | 148 | /* request callback tasklet */ |
149 | struct tasklet_struct done_task; | 149 | struct tasklet_struct done_task[2]; |
150 | 150 | ||
151 | /* list of registered algorithms */ | 151 | /* list of registered algorithms */ |
152 | struct list_head alg_list; | 152 | struct list_head alg_list; |
@@ -226,13 +226,19 @@ static int reset_device(struct device *dev) | |||
226 | { | 226 | { |
227 | struct talitos_private *priv = dev_get_drvdata(dev); | 227 | struct talitos_private *priv = dev_get_drvdata(dev); |
228 | unsigned int timeout = TALITOS_TIMEOUT; | 228 | unsigned int timeout = TALITOS_TIMEOUT; |
229 | u32 mcr = TALITOS_MCR_SWR; | ||
229 | 230 | ||
230 | setbits32(priv->reg + TALITOS_MCR, TALITOS_MCR_SWR); | 231 | setbits32(priv->reg + TALITOS_MCR, mcr); |
231 | 232 | ||
232 | while ((in_be32(priv->reg + TALITOS_MCR) & TALITOS_MCR_SWR) | 233 | while ((in_be32(priv->reg + TALITOS_MCR) & TALITOS_MCR_SWR) |
233 | && --timeout) | 234 | && --timeout) |
234 | cpu_relax(); | 235 | cpu_relax(); |
235 | 236 | ||
237 | if (priv->irq[1] != NO_IRQ) { | ||
238 | mcr = TALITOS_MCR_RCA1 | TALITOS_MCR_RCA3; | ||
239 | setbits32(priv->reg + TALITOS_MCR, mcr); | ||
240 | } | ||
241 | |||
236 | if (timeout == 0) { | 242 | if (timeout == 0) { |
237 | dev_err(dev, "failed to reset device\n"); | 243 | dev_err(dev, "failed to reset device\n"); |
238 | return -EIO; | 244 | return -EIO; |
@@ -401,21 +407,32 @@ static void flush_channel(struct device *dev, int ch, int error, int reset_ch) | |||
401 | /* | 407 | /* |
402 | * process completed requests for channels that have done status | 408 | * process completed requests for channels that have done status |
403 | */ | 409 | */ |
404 | static void talitos_done(unsigned long data) | 410 | #define DEF_TALITOS_DONE(name, ch_done_mask) \ |
405 | { | 411 | static void talitos_done_##name(unsigned long data) \ |
406 | struct device *dev = (struct device *)data; | 412 | { \ |
407 | struct talitos_private *priv = dev_get_drvdata(dev); | 413 | struct device *dev = (struct device *)data; \ |
408 | int ch; | 414 | struct talitos_private *priv = dev_get_drvdata(dev); \ |
409 | 415 | \ | |
410 | for (ch = 0; ch < priv->num_channels; ch++) | 416 | if (ch_done_mask & 1) \ |
411 | flush_channel(dev, ch, 0, 0); | 417 | flush_channel(dev, 0, 0, 0); \ |
412 | 418 | if (priv->num_channels == 1) \ | |
413 | /* At this point, all completed channels have been processed. | 419 | goto out; \ |
414 | * Unmask done interrupts for channels completed later on. | 420 | if (ch_done_mask & (1 << 2)) \ |
415 | */ | 421 | flush_channel(dev, 1, 0, 0); \ |
416 | setbits32(priv->reg + TALITOS_IMR, TALITOS_IMR_INIT); | 422 | if (ch_done_mask & (1 << 4)) \ |
417 | setbits32(priv->reg + TALITOS_IMR_LO, TALITOS_IMR_LO_INIT); | 423 | flush_channel(dev, 2, 0, 0); \ |
418 | } | 424 | if (ch_done_mask & (1 << 6)) \ |
425 | flush_channel(dev, 3, 0, 0); \ | ||
426 | \ | ||
427 | out: \ | ||
428 | /* At this point, all completed channels have been processed */ \ | ||
429 | /* Unmask done interrupts for channels completed later on. */ \ | ||
430 | setbits32(priv->reg + TALITOS_IMR, ch_done_mask); \ | ||
431 | setbits32(priv->reg + TALITOS_IMR_LO, TALITOS_IMR_LO_INIT); \ | ||
432 | } | ||
433 | DEF_TALITOS_DONE(4ch, TALITOS_ISR_4CHDONE) | ||
434 | DEF_TALITOS_DONE(ch0_2, TALITOS_ISR_CH_0_2_DONE) | ||
435 | DEF_TALITOS_DONE(ch1_3, TALITOS_ISR_CH_1_3_DONE) | ||
419 | 436 | ||
420 | /* | 437 | /* |
421 | * locate current (offending) descriptor | 438 | * locate current (offending) descriptor |
@@ -584,7 +601,7 @@ static void talitos_error(unsigned long data, u32 isr, u32 isr_lo) | |||
584 | } | 601 | } |
585 | } | 602 | } |
586 | } | 603 | } |
587 | if (reset_dev || isr & ~TALITOS_ISR_CHERR || isr_lo) { | 604 | if (reset_dev || isr & ~TALITOS_ISR_4CHERR || isr_lo) { |
588 | dev_err(dev, "done overflow, internal time out, or rngu error: " | 605 | dev_err(dev, "done overflow, internal time out, or rngu error: " |
589 | "ISR 0x%08x_%08x\n", isr, isr_lo); | 606 | "ISR 0x%08x_%08x\n", isr, isr_lo); |
590 | 607 | ||
@@ -597,30 +614,35 @@ static void talitos_error(unsigned long data, u32 isr, u32 isr_lo) | |||
597 | } | 614 | } |
598 | } | 615 | } |
599 | 616 | ||
600 | static irqreturn_t talitos_interrupt(int irq, void *data) | 617 | #define DEF_TALITOS_INTERRUPT(name, ch_done_mask, ch_err_mask, tlet) \ |
601 | { | 618 | static irqreturn_t talitos_interrupt_##name(int irq, void *data) \ |
602 | struct device *dev = data; | 619 | { \ |
603 | struct talitos_private *priv = dev_get_drvdata(dev); | 620 | struct device *dev = data; \ |
604 | u32 isr, isr_lo; | 621 | struct talitos_private *priv = dev_get_drvdata(dev); \ |
605 | 622 | u32 isr, isr_lo; \ | |
606 | isr = in_be32(priv->reg + TALITOS_ISR); | 623 | \ |
607 | isr_lo = in_be32(priv->reg + TALITOS_ISR_LO); | 624 | isr = in_be32(priv->reg + TALITOS_ISR); \ |
608 | /* Acknowledge interrupt */ | 625 | isr_lo = in_be32(priv->reg + TALITOS_ISR_LO); \ |
609 | out_be32(priv->reg + TALITOS_ICR, isr); | 626 | /* Acknowledge interrupt */ \ |
610 | out_be32(priv->reg + TALITOS_ICR_LO, isr_lo); | 627 | out_be32(priv->reg + TALITOS_ICR, isr & (ch_done_mask | ch_err_mask)); \ |
611 | 628 | out_be32(priv->reg + TALITOS_ICR_LO, isr_lo); \ | |
612 | if (unlikely((isr & ~TALITOS_ISR_CHDONE) || isr_lo)) | 629 | \ |
613 | talitos_error((unsigned long)data, isr, isr_lo); | 630 | if (unlikely((isr & ~TALITOS_ISR_4CHDONE) & ch_err_mask || isr_lo)) \ |
614 | else | 631 | talitos_error((unsigned long)data, isr, isr_lo); \ |
615 | if (likely(isr & TALITOS_ISR_CHDONE)) { | 632 | else \ |
616 | /* mask further done interrupts. */ | 633 | if (likely(isr & ch_done_mask)) { \ |
617 | clrbits32(priv->reg + TALITOS_IMR, TALITOS_IMR_DONE); | 634 | /* mask further done interrupts. */ \ |
618 | /* done_task will unmask done interrupts at exit */ | 635 | clrbits32(priv->reg + TALITOS_IMR, ch_done_mask); \ |
619 | tasklet_schedule(&priv->done_task); | 636 | /* done_task will unmask done interrupts at exit */ \ |
620 | } | 637 | tasklet_schedule(&priv->done_task[tlet]); \ |
621 | 638 | } \ | |
622 | return (isr || isr_lo) ? IRQ_HANDLED : IRQ_NONE; | 639 | \ |
623 | } | 640 | return (isr & (ch_done_mask | ch_err_mask) || isr_lo) ? IRQ_HANDLED : \ |
641 | IRQ_NONE; \ | ||
642 | } | ||
643 | DEF_TALITOS_INTERRUPT(4ch, TALITOS_ISR_4CHDONE, TALITOS_ISR_4CHERR, 0) | ||
644 | DEF_TALITOS_INTERRUPT(ch0_2, TALITOS_ISR_CH_0_2_DONE, TALITOS_ISR_CH_0_2_ERR, 0) | ||
645 | DEF_TALITOS_INTERRUPT(ch1_3, TALITOS_ISR_CH_1_3_DONE, TALITOS_ISR_CH_1_3_ERR, 1) | ||
624 | 646 | ||
625 | /* | 647 | /* |
626 | * hwrng | 648 | * hwrng |
@@ -2558,12 +2580,15 @@ static int talitos_remove(struct platform_device *ofdev) | |||
2558 | 2580 | ||
2559 | kfree(priv->chan); | 2581 | kfree(priv->chan); |
2560 | 2582 | ||
2561 | if (priv->irq != NO_IRQ) { | 2583 | for (i = 0; i < 2; i++) |
2562 | free_irq(priv->irq, dev); | 2584 | if (priv->irq[i] != NO_IRQ) { |
2563 | irq_dispose_mapping(priv->irq); | 2585 | free_irq(priv->irq[i], dev); |
2564 | } | 2586 | irq_dispose_mapping(priv->irq[i]); |
2587 | } | ||
2565 | 2588 | ||
2566 | tasklet_kill(&priv->done_task); | 2589 | tasklet_kill(&priv->done_task[0]); |
2590 | if (priv->irq[1] != NO_IRQ) | ||
2591 | tasklet_kill(&priv->done_task[1]); | ||
2567 | 2592 | ||
2568 | iounmap(priv->reg); | 2593 | iounmap(priv->reg); |
2569 | 2594 | ||
@@ -2628,6 +2653,54 @@ static struct talitos_crypto_alg *talitos_alg_alloc(struct device *dev, | |||
2628 | return t_alg; | 2653 | return t_alg; |
2629 | } | 2654 | } |
2630 | 2655 | ||
2656 | static int talitos_probe_irq(struct platform_device *ofdev) | ||
2657 | { | ||
2658 | struct device *dev = &ofdev->dev; | ||
2659 | struct device_node *np = ofdev->dev.of_node; | ||
2660 | struct talitos_private *priv = dev_get_drvdata(dev); | ||
2661 | int err; | ||
2662 | |||
2663 | priv->irq[0] = irq_of_parse_and_map(np, 0); | ||
2664 | if (priv->irq[0] == NO_IRQ) { | ||
2665 | dev_err(dev, "failed to map irq\n"); | ||
2666 | return -EINVAL; | ||
2667 | } | ||
2668 | |||
2669 | priv->irq[1] = irq_of_parse_and_map(np, 1); | ||
2670 | |||
2671 | /* get the primary irq line */ | ||
2672 | if (priv->irq[1] == NO_IRQ) { | ||
2673 | err = request_irq(priv->irq[0], talitos_interrupt_4ch, 0, | ||
2674 | dev_driver_string(dev), dev); | ||
2675 | goto primary_out; | ||
2676 | } | ||
2677 | |||
2678 | err = request_irq(priv->irq[0], talitos_interrupt_ch0_2, 0, | ||
2679 | dev_driver_string(dev), dev); | ||
2680 | if (err) | ||
2681 | goto primary_out; | ||
2682 | |||
2683 | /* get the secondary irq line */ | ||
2684 | err = request_irq(priv->irq[1], talitos_interrupt_ch1_3, 0, | ||
2685 | dev_driver_string(dev), dev); | ||
2686 | if (err) { | ||
2687 | dev_err(dev, "failed to request secondary irq\n"); | ||
2688 | irq_dispose_mapping(priv->irq[1]); | ||
2689 | priv->irq[1] = NO_IRQ; | ||
2690 | } | ||
2691 | |||
2692 | return err; | ||
2693 | |||
2694 | primary_out: | ||
2695 | if (err) { | ||
2696 | dev_err(dev, "failed to request primary irq\n"); | ||
2697 | irq_dispose_mapping(priv->irq[0]); | ||
2698 | priv->irq[0] = NO_IRQ; | ||
2699 | } | ||
2700 | |||
2701 | return err; | ||
2702 | } | ||
2703 | |||
2631 | static int talitos_probe(struct platform_device *ofdev) | 2704 | static int talitos_probe(struct platform_device *ofdev) |
2632 | { | 2705 | { |
2633 | struct device *dev = &ofdev->dev; | 2706 | struct device *dev = &ofdev->dev; |
@@ -2644,28 +2717,22 @@ static int talitos_probe(struct platform_device *ofdev) | |||
2644 | 2717 | ||
2645 | priv->ofdev = ofdev; | 2718 | priv->ofdev = ofdev; |
2646 | 2719 | ||
2647 | tasklet_init(&priv->done_task, talitos_done, (unsigned long)dev); | 2720 | err = talitos_probe_irq(ofdev); |
2648 | 2721 | if (err) | |
2649 | INIT_LIST_HEAD(&priv->alg_list); | ||
2650 | |||
2651 | priv->irq = irq_of_parse_and_map(np, 0); | ||
2652 | |||
2653 | if (priv->irq == NO_IRQ) { | ||
2654 | dev_err(dev, "failed to map irq\n"); | ||
2655 | err = -EINVAL; | ||
2656 | goto err_out; | 2722 | goto err_out; |
2657 | } | ||
2658 | 2723 | ||
2659 | /* get the irq line */ | 2724 | if (priv->irq[1] == NO_IRQ) { |
2660 | err = request_irq(priv->irq, talitos_interrupt, 0, | 2725 | tasklet_init(&priv->done_task[0], talitos_done_4ch, |
2661 | dev_driver_string(dev), dev); | 2726 | (unsigned long)dev); |
2662 | if (err) { | 2727 | } else { |
2663 | dev_err(dev, "failed to request irq %d\n", priv->irq); | 2728 | tasklet_init(&priv->done_task[0], talitos_done_ch0_2, |
2664 | irq_dispose_mapping(priv->irq); | 2729 | (unsigned long)dev); |
2665 | priv->irq = NO_IRQ; | 2730 | tasklet_init(&priv->done_task[1], talitos_done_ch1_3, |
2666 | goto err_out; | 2731 | (unsigned long)dev); |
2667 | } | 2732 | } |
2668 | 2733 | ||
2734 | INIT_LIST_HEAD(&priv->alg_list); | ||
2735 | |||
2669 | priv->reg = of_iomap(np, 0); | 2736 | priv->reg = of_iomap(np, 0); |
2670 | if (!priv->reg) { | 2737 | if (!priv->reg) { |
2671 | dev_err(dev, "failed to of_iomap\n"); | 2738 | dev_err(dev, "failed to of_iomap\n"); |
@@ -2713,9 +2780,11 @@ static int talitos_probe(struct platform_device *ofdev) | |||
2713 | goto err_out; | 2780 | goto err_out; |
2714 | } | 2781 | } |
2715 | 2782 | ||
2716 | for (i = 0; i < priv->num_channels; i++) | 2783 | for (i = 0; i < priv->num_channels; i++) { |
2717 | priv->chan[i].reg = priv->reg + TALITOS_CH_BASE_OFFSET + | 2784 | priv->chan[i].reg = priv->reg + TALITOS_CH_STRIDE * (i + 1); |
2718 | TALITOS_CH_STRIDE * (i + 1); | 2785 | if ((priv->irq[1] == NO_IRQ) || !(i & 1)) |
2786 | priv->chan[i].reg += TALITOS_CH_BASE_OFFSET; | ||
2787 | } | ||
2719 | 2788 | ||
2720 | for (i = 0; i < priv->num_channels; i++) { | 2789 | for (i = 0; i < priv->num_channels; i++) { |
2721 | spin_lock_init(&priv->chan[i].head_lock); | 2790 | spin_lock_init(&priv->chan[i].head_lock); |