148 lines
3.5 KiB
C
148 lines
3.5 KiB
C
/*
|
|
* Copyright (C) 2010 Google, Inc.
|
|
*
|
|
* This software is licensed under the terms of the GNU General Public
|
|
* License version 2, as published by the Free Software Foundation, and
|
|
* may be copied, distributed, and modified under those terms.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/percpu.h>
|
|
#include <linux/slab.h>
|
|
#include <asm/fiq.h>
|
|
#include <asm/fiq_glue.h>
|
|
|
|
extern unsigned char fiq_glue, fiq_glue_end;
|
|
extern void fiq_glue_setup(void *func, void *data, void *sp,
|
|
fiq_return_handler_t fiq_return_handler);
|
|
|
|
static struct fiq_handler fiq_debbuger_fiq_handler = {
|
|
.name = "fiq_glue",
|
|
};
|
|
DEFINE_PER_CPU(void *, fiq_stack);
|
|
static struct fiq_glue_handler *current_handler;
|
|
static fiq_return_handler_t fiq_return_handler;
|
|
static DEFINE_MUTEX(fiq_glue_lock);
|
|
|
|
static void fiq_glue_setup_helper(void *info)
|
|
{
|
|
struct fiq_glue_handler *handler = info;
|
|
fiq_glue_setup(handler->fiq, handler,
|
|
__get_cpu_var(fiq_stack) + THREAD_START_SP,
|
|
fiq_return_handler);
|
|
}
|
|
|
|
int fiq_glue_register_handler(struct fiq_glue_handler *handler)
|
|
{
|
|
int ret;
|
|
int cpu;
|
|
|
|
if (!handler || !handler->fiq)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&fiq_glue_lock);
|
|
if (fiq_stack) {
|
|
ret = -EBUSY;
|
|
goto err_busy;
|
|
}
|
|
|
|
for_each_possible_cpu(cpu) {
|
|
void *stack;
|
|
stack = (void *)__get_free_pages(GFP_KERNEL, THREAD_SIZE_ORDER);
|
|
if (WARN_ON(!stack)) {
|
|
ret = -ENOMEM;
|
|
goto err_alloc_fiq_stack;
|
|
}
|
|
per_cpu(fiq_stack, cpu) = stack;
|
|
}
|
|
|
|
ret = claim_fiq(&fiq_debbuger_fiq_handler);
|
|
if (WARN_ON(ret))
|
|
goto err_claim_fiq;
|
|
|
|
current_handler = handler;
|
|
on_each_cpu(fiq_glue_setup_helper, handler, true);
|
|
set_fiq_handler(&fiq_glue, &fiq_glue_end - &fiq_glue);
|
|
|
|
mutex_unlock(&fiq_glue_lock);
|
|
return 0;
|
|
|
|
err_claim_fiq:
|
|
err_alloc_fiq_stack:
|
|
for_each_possible_cpu(cpu) {
|
|
__free_pages(per_cpu(fiq_stack, cpu), THREAD_SIZE_ORDER);
|
|
per_cpu(fiq_stack, cpu) = NULL;
|
|
}
|
|
err_busy:
|
|
mutex_unlock(&fiq_glue_lock);
|
|
return ret;
|
|
}
|
|
|
|
static void fiq_glue_update_return_handler(void (*fiq_return)(void))
|
|
{
|
|
fiq_return_handler = fiq_return;
|
|
if (current_handler)
|
|
on_each_cpu(fiq_glue_setup_helper, current_handler, true);
|
|
}
|
|
|
|
int fiq_glue_set_return_handler(void (*fiq_return)(void))
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock(&fiq_glue_lock);
|
|
if (fiq_return_handler) {
|
|
ret = -EBUSY;
|
|
goto err_busy;
|
|
}
|
|
fiq_glue_update_return_handler(fiq_return);
|
|
ret = 0;
|
|
err_busy:
|
|
mutex_unlock(&fiq_glue_lock);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL(fiq_glue_set_return_handler);
|
|
|
|
int fiq_glue_clear_return_handler(void (*fiq_return)(void))
|
|
{
|
|
int ret;
|
|
|
|
mutex_lock(&fiq_glue_lock);
|
|
if (WARN_ON(fiq_return_handler != fiq_return)) {
|
|
ret = -EINVAL;
|
|
goto err_inval;
|
|
}
|
|
fiq_glue_update_return_handler(NULL);
|
|
ret = 0;
|
|
err_inval:
|
|
mutex_unlock(&fiq_glue_lock);
|
|
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL(fiq_glue_clear_return_handler);
|
|
|
|
/**
|
|
* fiq_glue_resume - Restore fiqs after suspend or low power idle states
|
|
*
|
|
* This must be called before calling local_fiq_enable after returning from a
|
|
* power state where the fiq mode registers were lost. If a driver provided
|
|
* a resume hook when it registered the handler it will be called.
|
|
*/
|
|
|
|
void fiq_glue_resume(void)
|
|
{
|
|
if (!current_handler)
|
|
return;
|
|
fiq_glue_setup(current_handler->fiq, current_handler,
|
|
__get_cpu_var(fiq_stack) + THREAD_START_SP,
|
|
fiq_return_handler);
|
|
if (current_handler->resume)
|
|
current_handler->resume(current_handler);
|
|
}
|
|
|