2006-06-27 23:03:03 +01:00
/*
* arch / arm / kernel / crunch . c
* Cirrus MaverickCrunch context switching and handling
*
* Copyright ( C ) 2006 Lennert Buytenhek < buytenh @ wantstofly . org >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/signal.h>
# include <linux/sched.h>
# include <linux/init.h>
2008-09-06 12:10:45 +01:00
# include <linux/io.h>
2012-01-11 13:14:20 +11:00
2006-06-27 23:03:03 +01:00
# include <asm/thread_notify.h>
2012-01-11 12:53:33 +11:00
# include "soc.h"
2006-06-27 23:03:03 +01:00
struct crunch_state * crunch_owner ;
void crunch_task_release ( struct thread_info * thread )
{
local_irq_disable ( ) ;
if ( crunch_owner = = & thread - > crunchstate )
crunch_owner = NULL ;
local_irq_enable ( ) ;
}
static int crunch_enabled ( u32 devcfg )
{
2009-07-08 02:00:49 +01:00
return ! ! ( devcfg & EP93XX_SYSCON_DEVCFG_CPENA ) ;
2006-06-27 23:03:03 +01:00
}
static int crunch_do ( struct notifier_block * self , unsigned long cmd , void * t )
{
struct thread_info * thread = ( struct thread_info * ) t ;
struct crunch_state * crunch_state ;
u32 devcfg ;
crunch_state = & thread - > crunchstate ;
switch ( cmd ) {
case THREAD_NOTIFY_FLUSH :
memset ( crunch_state , 0 , sizeof ( * crunch_state ) ) ;
/*
* FALLTHROUGH : Ensure we don ' t try to overwrite our newly
* initialised state information on the first fault .
*/
2009-12-18 14:34:43 +00:00
case THREAD_NOTIFY_EXIT :
2006-06-27 23:03:03 +01:00
crunch_task_release ( thread ) ;
break ;
case THREAD_NOTIFY_SWITCH :
2009-07-08 02:00:49 +01:00
devcfg = __raw_readl ( EP93XX_SYSCON_DEVCFG ) ;
2006-06-27 23:03:03 +01:00
if ( crunch_enabled ( devcfg ) | | crunch_owner = = crunch_state ) {
2009-07-08 02:00:49 +01:00
/*
* We don ' t use ep93xx_syscon_swlocked_write ( ) here
* because we are on the context switch path and
* preemption is already disabled .
*/
devcfg ^ = EP93XX_SYSCON_DEVCFG_CPENA ;
2006-06-27 23:03:03 +01:00
__raw_writel ( 0xaa , EP93XX_SYSCON_SWLOCK ) ;
2009-07-08 02:00:49 +01:00
__raw_writel ( devcfg , EP93XX_SYSCON_DEVCFG ) ;
2006-06-27 23:03:03 +01:00
}
break ;
}
return NOTIFY_DONE ;
}
static struct notifier_block crunch_notifier_block = {
. notifier_call = crunch_do ,
} ;
2012-04-26 10:05:15 +08:00
int __init crunch_init ( void )
2006-06-27 23:03:03 +01:00
{
thread_register_notifier ( & crunch_notifier_block ) ;
2007-02-05 00:35:37 +01:00
elf_hwcap | = HWCAP_CRUNCH ;
2006-06-27 23:03:03 +01:00
return 0 ;
}