mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	Move from the old, complex intr handling code to a new implementation based on dma_fences. While there is a fair bit of churn to get there, the new implementation is much simpler and likely faster as well due to allowing signaling directly from interrupt context. Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> Signed-off-by: Thierry Reding <treding@nvidia.com>
		
			
				
	
	
		
			138 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0-only
 | 
						|
/*
 | 
						|
 * Tegra host1x Interrupt Management
 | 
						|
 *
 | 
						|
 * Copyright (c) 2010-2021, NVIDIA Corporation.
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/clk.h>
 | 
						|
 | 
						|
#include "dev.h"
 | 
						|
#include "fence.h"
 | 
						|
#include "intr.h"
 | 
						|
 | 
						|
static void host1x_intr_add_fence_to_list(struct host1x_fence_list *list,
 | 
						|
					  struct host1x_syncpt_fence *fence)
 | 
						|
{
 | 
						|
	struct host1x_syncpt_fence *fence_in_list;
 | 
						|
 | 
						|
	list_for_each_entry_reverse(fence_in_list, &list->list, list) {
 | 
						|
		if ((s32)(fence_in_list->threshold - fence->threshold) <= 0) {
 | 
						|
			/* Fence in list is before us, we can insert here */
 | 
						|
			list_add(&fence->list, &fence_in_list->list);
 | 
						|
			return;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Add as first in list */
 | 
						|
	list_add(&fence->list, &list->list);
 | 
						|
}
 | 
						|
 | 
						|
static void host1x_intr_update_hw_state(struct host1x *host, struct host1x_syncpt *sp)
 | 
						|
{
 | 
						|
	struct host1x_syncpt_fence *fence;
 | 
						|
 | 
						|
	if (!list_empty(&sp->fences.list)) {
 | 
						|
		fence = list_first_entry(&sp->fences.list, struct host1x_syncpt_fence, list);
 | 
						|
 | 
						|
		host1x_hw_intr_set_syncpt_threshold(host, sp->id, fence->threshold);
 | 
						|
		host1x_hw_intr_enable_syncpt_intr(host, sp->id);
 | 
						|
	} else {
 | 
						|
		host1x_hw_intr_disable_syncpt_intr(host, sp->id);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void host1x_intr_add_fence_locked(struct host1x *host, struct host1x_syncpt_fence *fence)
 | 
						|
{
 | 
						|
	struct host1x_fence_list *fence_list = &fence->sp->fences;
 | 
						|
 | 
						|
	INIT_LIST_HEAD(&fence->list);
 | 
						|
 | 
						|
	host1x_intr_add_fence_to_list(fence_list, fence);
 | 
						|
	host1x_intr_update_hw_state(host, fence->sp);
 | 
						|
}
 | 
						|
 | 
						|
bool host1x_intr_remove_fence(struct host1x *host, struct host1x_syncpt_fence *fence)
 | 
						|
{
 | 
						|
	struct host1x_fence_list *fence_list = &fence->sp->fences;
 | 
						|
	unsigned long irqflags;
 | 
						|
 | 
						|
	spin_lock_irqsave(&fence_list->lock, irqflags);
 | 
						|
 | 
						|
	if (list_empty(&fence->list)) {
 | 
						|
		spin_unlock_irqrestore(&fence_list->lock, irqflags);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	list_del_init(&fence->list);
 | 
						|
	host1x_intr_update_hw_state(host, fence->sp);
 | 
						|
 | 
						|
	spin_unlock_irqrestore(&fence_list->lock, irqflags);
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void host1x_intr_handle_interrupt(struct host1x *host, unsigned int id)
 | 
						|
{
 | 
						|
	struct host1x_syncpt *sp = &host->syncpt[id];
 | 
						|
	struct host1x_syncpt_fence *fence, *tmp;
 | 
						|
	unsigned int value;
 | 
						|
 | 
						|
	value = host1x_syncpt_load(sp);
 | 
						|
 | 
						|
	spin_lock(&sp->fences.lock);
 | 
						|
 | 
						|
	list_for_each_entry_safe(fence, tmp, &sp->fences.list, list) {
 | 
						|
		if (((value - fence->threshold) & 0x80000000U) != 0U) {
 | 
						|
			/* Fence is not yet expired, we are done */
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		list_del_init(&fence->list);
 | 
						|
		host1x_fence_signal(fence);
 | 
						|
	}
 | 
						|
 | 
						|
	/* Re-enable interrupt if necessary */
 | 
						|
	host1x_intr_update_hw_state(host, sp);
 | 
						|
 | 
						|
	spin_unlock(&sp->fences.lock);
 | 
						|
}
 | 
						|
 | 
						|
int host1x_intr_init(struct host1x *host)
 | 
						|
{
 | 
						|
	unsigned int id;
 | 
						|
 | 
						|
	mutex_init(&host->intr_mutex);
 | 
						|
 | 
						|
	for (id = 0; id < host1x_syncpt_nb_pts(host); ++id) {
 | 
						|
		struct host1x_syncpt *syncpt = &host->syncpt[id];
 | 
						|
 | 
						|
		spin_lock_init(&syncpt->fences.lock);
 | 
						|
		INIT_LIST_HEAD(&syncpt->fences.list);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void host1x_intr_deinit(struct host1x *host)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
void host1x_intr_start(struct host1x *host)
 | 
						|
{
 | 
						|
	u32 hz = clk_get_rate(host->clk);
 | 
						|
	int err;
 | 
						|
 | 
						|
	mutex_lock(&host->intr_mutex);
 | 
						|
	err = host1x_hw_intr_init_host_sync(host, DIV_ROUND_UP(hz, 1000000));
 | 
						|
	if (err) {
 | 
						|
		mutex_unlock(&host->intr_mutex);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	mutex_unlock(&host->intr_mutex);
 | 
						|
}
 | 
						|
 | 
						|
void host1x_intr_stop(struct host1x *host)
 | 
						|
{
 | 
						|
	host1x_hw_intr_disable_all_syncpt_intrs(host);
 | 
						|
}
 |