Lines Matching full:channel

191 static void rz_dmac_ch_writel(struct rz_dmac_chan *channel, unsigned int val,  in rz_dmac_ch_writel()  argument
195 writel(val, channel->ch_base + offset); in rz_dmac_ch_writel()
197 writel(val, channel->ch_cmn_base + offset); in rz_dmac_ch_writel()
200 static u32 rz_dmac_ch_readl(struct rz_dmac_chan *channel, in rz_dmac_ch_readl() argument
204 return readl(channel->ch_base + offset); in rz_dmac_ch_readl()
206 return readl(channel->ch_cmn_base + offset); in rz_dmac_ch_readl()
214 static void rz_lmdesc_setup(struct rz_dmac_chan *channel, in rz_lmdesc_setup() argument
219 channel->lmdesc.base = lmdesc; in rz_lmdesc_setup()
220 channel->lmdesc.head = lmdesc; in rz_lmdesc_setup()
221 channel->lmdesc.tail = lmdesc; in rz_lmdesc_setup()
222 nxla = channel->lmdesc.base_dma; in rz_lmdesc_setup()
223 while (lmdesc < (channel->lmdesc.base + (DMAC_NR_LMDESC - 1))) { in rz_lmdesc_setup()
231 lmdesc->nxla = channel->lmdesc.base_dma; in rz_lmdesc_setup()
239 static void rz_dmac_lmdesc_recycle(struct rz_dmac_chan *channel) in rz_dmac_lmdesc_recycle() argument
241 struct rz_lmdesc *lmdesc = channel->lmdesc.head; in rz_dmac_lmdesc_recycle()
246 if (lmdesc >= (channel->lmdesc.base + DMAC_NR_LMDESC)) in rz_dmac_lmdesc_recycle()
247 lmdesc = channel->lmdesc.base; in rz_dmac_lmdesc_recycle()
249 channel->lmdesc.head = lmdesc; in rz_dmac_lmdesc_recycle()
252 static void rz_dmac_enable_hw(struct rz_dmac_chan *channel) in rz_dmac_enable_hw() argument
254 struct dma_chan *chan = &channel->vc.chan; in rz_dmac_enable_hw()
261 dev_dbg(dmac->dev, "%s channel %d\n", __func__, channel->index); in rz_dmac_enable_hw()
265 rz_dmac_lmdesc_recycle(channel); in rz_dmac_enable_hw()
267 nxla = channel->lmdesc.base_dma + in rz_dmac_enable_hw()
268 (sizeof(struct rz_lmdesc) * (channel->lmdesc.head - in rz_dmac_enable_hw()
269 channel->lmdesc.base)); in rz_dmac_enable_hw()
271 chstat = rz_dmac_ch_readl(channel, CHSTAT, 1); in rz_dmac_enable_hw()
273 chctrl = (channel->chctrl | CHCTRL_SETEN); in rz_dmac_enable_hw()
274 rz_dmac_ch_writel(channel, nxla, NXLA, 1); in rz_dmac_enable_hw()
275 rz_dmac_ch_writel(channel, channel->chcfg, CHCFG, 1); in rz_dmac_enable_hw()
276 rz_dmac_ch_writel(channel, CHCTRL_SWRST, CHCTRL, 1); in rz_dmac_enable_hw()
277 rz_dmac_ch_writel(channel, chctrl, CHCTRL, 1); in rz_dmac_enable_hw()
283 static void rz_dmac_disable_hw(struct rz_dmac_chan *channel) in rz_dmac_disable_hw() argument
285 struct dma_chan *chan = &channel->vc.chan; in rz_dmac_disable_hw()
289 dev_dbg(dmac->dev, "%s channel %d\n", __func__, channel->index); in rz_dmac_disable_hw()
292 rz_dmac_ch_writel(channel, CHCTRL_DEFAULT, CHCTRL, 1); in rz_dmac_disable_hw()
309 static void rz_dmac_prepare_desc_for_memcpy(struct rz_dmac_chan *channel) in rz_dmac_prepare_desc_for_memcpy() argument
311 struct dma_chan *chan = &channel->vc.chan; in rz_dmac_prepare_desc_for_memcpy()
313 struct rz_lmdesc *lmdesc = channel->lmdesc.tail; in rz_dmac_prepare_desc_for_memcpy()
314 struct rz_dmac_desc *d = channel->desc; in rz_dmac_prepare_desc_for_memcpy()
326 rz_dmac_set_dmars_register(dmac, channel->index, 0); in rz_dmac_prepare_desc_for_memcpy()
328 channel->chcfg = chcfg; in rz_dmac_prepare_desc_for_memcpy()
329 channel->chctrl = CHCTRL_STG | CHCTRL_SETEN; in rz_dmac_prepare_desc_for_memcpy()
332 static void rz_dmac_prepare_descs_for_slave_sg(struct rz_dmac_chan *channel) in rz_dmac_prepare_descs_for_slave_sg() argument
334 struct dma_chan *chan = &channel->vc.chan; in rz_dmac_prepare_descs_for_slave_sg()
336 struct rz_dmac_desc *d = channel->desc; in rz_dmac_prepare_descs_for_slave_sg()
341 channel->chcfg |= CHCFG_SEL(channel->index) | CHCFG_DEM | CHCFG_DMS; in rz_dmac_prepare_descs_for_slave_sg()
344 channel->chcfg |= CHCFG_SAD; in rz_dmac_prepare_descs_for_slave_sg()
345 channel->chcfg &= ~CHCFG_REQD; in rz_dmac_prepare_descs_for_slave_sg()
347 channel->chcfg |= CHCFG_DAD | CHCFG_REQD; in rz_dmac_prepare_descs_for_slave_sg()
350 lmdesc = channel->lmdesc.tail; in rz_dmac_prepare_descs_for_slave_sg()
354 lmdesc->sa = channel->src_per_address; in rz_dmac_prepare_descs_for_slave_sg()
358 lmdesc->da = channel->dst_per_address; in rz_dmac_prepare_descs_for_slave_sg()
365 lmdesc->chcfg = (channel->chcfg & ~CHCFG_DEM); in rz_dmac_prepare_descs_for_slave_sg()
368 lmdesc->chcfg = channel->chcfg; in rz_dmac_prepare_descs_for_slave_sg()
371 if (++lmdesc >= (channel->lmdesc.base + DMAC_NR_LMDESC)) in rz_dmac_prepare_descs_for_slave_sg()
372 lmdesc = channel->lmdesc.base; in rz_dmac_prepare_descs_for_slave_sg()
375 channel->lmdesc.tail = lmdesc; in rz_dmac_prepare_descs_for_slave_sg()
377 rz_dmac_set_dmars_register(dmac, channel->index, channel->mid_rid); in rz_dmac_prepare_descs_for_slave_sg()
378 channel->chctrl = CHCTRL_SETEN; in rz_dmac_prepare_descs_for_slave_sg()
417 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_alloc_chan_resources() local
419 while (channel->descs_allocated < RZ_DMAC_MAX_CHAN_DESCRIPTORS) { in rz_dmac_alloc_chan_resources()
426 list_add_tail(&desc->node, &channel->ld_free); in rz_dmac_alloc_chan_resources()
427 channel->descs_allocated++; in rz_dmac_alloc_chan_resources()
430 if (!channel->descs_allocated) in rz_dmac_alloc_chan_resources()
433 return channel->descs_allocated; in rz_dmac_alloc_chan_resources()
438 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_free_chan_resources() local
440 struct rz_lmdesc *lmdesc = channel->lmdesc.base; in rz_dmac_free_chan_resources()
445 spin_lock_irqsave(&channel->vc.lock, flags); in rz_dmac_free_chan_resources()
450 rz_dmac_disable_hw(channel); in rz_dmac_free_chan_resources()
451 list_splice_tail_init(&channel->ld_active, &channel->ld_free); in rz_dmac_free_chan_resources()
452 list_splice_tail_init(&channel->ld_queue, &channel->ld_free); in rz_dmac_free_chan_resources()
454 if (channel->mid_rid >= 0) { in rz_dmac_free_chan_resources()
455 clear_bit(channel->mid_rid, dmac->modules); in rz_dmac_free_chan_resources()
456 channel->mid_rid = -EINVAL; in rz_dmac_free_chan_resources()
459 spin_unlock_irqrestore(&channel->vc.lock, flags); in rz_dmac_free_chan_resources()
461 list_for_each_entry_safe(desc, _desc, &channel->ld_free, node) { in rz_dmac_free_chan_resources()
463 channel->descs_allocated--; in rz_dmac_free_chan_resources()
466 INIT_LIST_HEAD(&channel->ld_free); in rz_dmac_free_chan_resources()
467 vchan_free_chan_resources(&channel->vc); in rz_dmac_free_chan_resources()
474 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_prep_dma_memcpy() local
478 dev_dbg(dmac->dev, "%s channel: %d src=0x%pad dst=0x%pad len=%zu\n", in rz_dmac_prep_dma_memcpy()
479 __func__, channel->index, &src, &dest, len); in rz_dmac_prep_dma_memcpy()
481 if (list_empty(&channel->ld_free)) in rz_dmac_prep_dma_memcpy()
484 desc = list_first_entry(&channel->ld_free, struct rz_dmac_desc, node); in rz_dmac_prep_dma_memcpy()
492 list_move_tail(channel->ld_free.next, &channel->ld_queue); in rz_dmac_prep_dma_memcpy()
493 return vchan_tx_prep(&channel->vc, &desc->vd, flags); in rz_dmac_prep_dma_memcpy()
502 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_prep_slave_sg() local
508 if (list_empty(&channel->ld_free)) in rz_dmac_prep_slave_sg()
511 desc = list_first_entry(&channel->ld_free, struct rz_dmac_desc, node); in rz_dmac_prep_slave_sg()
524 desc->src = channel->src_per_address; in rz_dmac_prep_slave_sg()
526 desc->dest = channel->dst_per_address; in rz_dmac_prep_slave_sg()
528 list_move_tail(channel->ld_free.next, &channel->ld_queue); in rz_dmac_prep_slave_sg()
529 return vchan_tx_prep(&channel->vc, &desc->vd, flags); in rz_dmac_prep_slave_sg()
534 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_terminate_all() local
538 rz_dmac_disable_hw(channel); in rz_dmac_terminate_all()
539 spin_lock_irqsave(&channel->vc.lock, flags); in rz_dmac_terminate_all()
540 list_splice_tail_init(&channel->ld_active, &channel->ld_free); in rz_dmac_terminate_all()
541 list_splice_tail_init(&channel->ld_queue, &channel->ld_free); in rz_dmac_terminate_all()
542 spin_unlock_irqrestore(&channel->vc.lock, flags); in rz_dmac_terminate_all()
543 vchan_get_all_descriptors(&channel->vc, &head); in rz_dmac_terminate_all()
544 vchan_dma_desc_free_list(&channel->vc, &head); in rz_dmac_terminate_all()
551 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_issue_pending() local
556 spin_lock_irqsave(&channel->vc.lock, flags); in rz_dmac_issue_pending()
558 if (!list_empty(&channel->ld_queue)) { in rz_dmac_issue_pending()
559 desc = list_first_entry(&channel->ld_queue, in rz_dmac_issue_pending()
561 channel->desc = desc; in rz_dmac_issue_pending()
562 if (vchan_issue_pending(&channel->vc)) { in rz_dmac_issue_pending()
563 if (rz_dmac_xfer_desc(channel) < 0) in rz_dmac_issue_pending()
565 channel->index); in rz_dmac_issue_pending()
567 list_move_tail(channel->ld_queue.next, in rz_dmac_issue_pending()
568 &channel->ld_active); in rz_dmac_issue_pending()
572 spin_unlock_irqrestore(&channel->vc.lock, flags); in rz_dmac_issue_pending()
600 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_config() local
603 channel->src_per_address = config->src_addr; in rz_dmac_config()
604 channel->src_word_size = config->src_addr_width; in rz_dmac_config()
605 channel->dst_per_address = config->dst_addr; in rz_dmac_config()
606 channel->dst_word_size = config->dst_addr_width; in rz_dmac_config()
612 channel->chcfg |= CHCFG_FILL_DDS(val); in rz_dmac_config()
618 channel->chcfg |= CHCFG_FILL_SDS(val); in rz_dmac_config()
636 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_device_synchronize() local
642 100, 100000, false, channel, CHSTAT, 1); in rz_dmac_device_synchronize()
646 rz_dmac_set_dmars_register(dmac, channel->index, 0); in rz_dmac_device_synchronize()
654 static void rz_dmac_irq_handle_channel(struct rz_dmac_chan *channel) in rz_dmac_irq_handle_channel() argument
656 struct dma_chan *chan = &channel->vc.chan; in rz_dmac_irq_handle_channel()
660 chstat = rz_dmac_ch_readl(channel, CHSTAT, 1); in rz_dmac_irq_handle_channel()
663 channel->index, chstat); in rz_dmac_irq_handle_channel()
664 rz_dmac_ch_writel(channel, CHCTRL_DEFAULT, CHCTRL, 1); in rz_dmac_irq_handle_channel()
668 chctrl = rz_dmac_ch_readl(channel, CHCTRL, 1); in rz_dmac_irq_handle_channel()
669 rz_dmac_ch_writel(channel, chctrl | CHCTRL_CLREND, CHCTRL, 1); in rz_dmac_irq_handle_channel()
676 struct rz_dmac_chan *channel = dev_id; in rz_dmac_irq_handler() local
678 if (channel) { in rz_dmac_irq_handler()
679 rz_dmac_irq_handle_channel(channel); in rz_dmac_irq_handler()
688 struct rz_dmac_chan *channel = dev_id; in rz_dmac_irq_handler_thread() local
692 spin_lock_irqsave(&channel->vc.lock, flags); in rz_dmac_irq_handler_thread()
694 if (list_empty(&channel->ld_active)) { in rz_dmac_irq_handler_thread()
699 desc = list_first_entry(&channel->ld_active, struct rz_dmac_desc, node); in rz_dmac_irq_handler_thread()
701 list_move_tail(channel->ld_active.next, &channel->ld_free); in rz_dmac_irq_handler_thread()
702 if (!list_empty(&channel->ld_queue)) { in rz_dmac_irq_handler_thread()
703 desc = list_first_entry(&channel->ld_queue, struct rz_dmac_desc, in rz_dmac_irq_handler_thread()
705 channel->desc = desc; in rz_dmac_irq_handler_thread()
706 if (rz_dmac_xfer_desc(channel) == 0) in rz_dmac_irq_handler_thread()
707 list_move_tail(channel->ld_queue.next, &channel->ld_active); in rz_dmac_irq_handler_thread()
710 spin_unlock_irqrestore(&channel->vc.lock, flags); in rz_dmac_irq_handler_thread()
717 * OF xlate and channel filter
722 struct rz_dmac_chan *channel = to_rz_dmac_chan(chan); in rz_dmac_chan_filter() local
727 channel->mid_rid = dma_spec->args[0] & MID_RID_MASK; in rz_dmac_chan_filter()
729 channel->chcfg = CHCFG_FILL_TM(ch_cfg) | CHCFG_FILL_AM(ch_cfg) | in rz_dmac_chan_filter()
732 return !test_and_set_bit(channel->mid_rid, dmac->modules); in rz_dmac_chan_filter()
756 struct rz_dmac_chan *channel, in rz_dmac_chan_probe() argument
765 channel->index = index; in rz_dmac_chan_probe()
766 channel->mid_rid = -EINVAL; in rz_dmac_chan_probe()
768 /* Request the channel interrupt. */ in rz_dmac_chan_probe()
770 channel->irq = platform_get_irq_byname(pdev, pdev_irqname); in rz_dmac_chan_probe()
771 if (channel->irq < 0) in rz_dmac_chan_probe()
772 return channel->irq; in rz_dmac_chan_probe()
779 ret = devm_request_threaded_irq(dmac->dev, channel->irq, in rz_dmac_chan_probe()
782 irqname, channel); in rz_dmac_chan_probe()
785 channel->irq, ret); in rz_dmac_chan_probe()
789 /* Set io base address for each channel */ in rz_dmac_chan_probe()
791 channel->ch_base = dmac->base + CHANNEL_0_7_OFFSET + in rz_dmac_chan_probe()
793 channel->ch_cmn_base = dmac->base + CHANNEL_0_7_COMMON_BASE; in rz_dmac_chan_probe()
795 channel->ch_base = dmac->base + CHANNEL_8_15_OFFSET + in rz_dmac_chan_probe()
797 channel->ch_cmn_base = dmac->base + CHANNEL_8_15_COMMON_BASE; in rz_dmac_chan_probe()
803 &channel->lmdesc.base_dma, GFP_KERNEL); in rz_dmac_chan_probe()
808 rz_lmdesc_setup(channel, lmdesc); in rz_dmac_chan_probe()
810 /* Initialize register for each channel */ in rz_dmac_chan_probe()
811 rz_dmac_ch_writel(channel, CHCTRL_DEFAULT, CHCTRL, 1); in rz_dmac_chan_probe()
813 channel->vc.desc_free = rz_dmac_virt_desc_free; in rz_dmac_chan_probe()
814 vchan_init(&channel->vc, &dmac->engine); in rz_dmac_chan_probe()
815 INIT_LIST_HEAD(&channel->ld_queue); in rz_dmac_chan_probe()
816 INIT_LIST_HEAD(&channel->ld_free); in rz_dmac_chan_probe()
817 INIT_LIST_HEAD(&channel->ld_active); in rz_dmac_chan_probe()
945 struct rz_dmac_chan *channel = &dmac->channels[i]; in rz_dmac_probe() local
949 channel->lmdesc.base, in rz_dmac_probe()
950 channel->lmdesc.base_dma); in rz_dmac_probe()
966 struct rz_dmac_chan *channel = &dmac->channels[i]; in rz_dmac_remove() local
970 channel->lmdesc.base, in rz_dmac_remove()
971 channel->lmdesc.base_dma); in rz_dmac_remove()