2010-05-17 10:12:46 +09:00
/*
* security / tomoyo / memory . c
*
2011-07-14 14:46:51 +09:00
* Copyright ( C ) 2005 - 2011 NTT DATA CORPORATION
2010-05-17 10:12:46 +09:00
*/
# include <linux/hash.h>
# include <linux/slab.h>
# include "common.h"
/**
* tomoyo_warn_oom - Print out of memory warning message .
*
* @ function : Function ' s name .
*/
void tomoyo_warn_oom ( const char * function )
{
/* Reduce error messages. */
static pid_t tomoyo_last_pid ;
const pid_t pid = current - > pid ;
if ( tomoyo_last_pid ! = pid ) {
printk ( KERN_WARNING " ERROR: Out of memory at %s. \n " ,
function ) ;
tomoyo_last_pid = pid ;
}
if ( ! tomoyo_policy_loaded )
panic ( " MAC Initialization failed. \n " ) ;
}
2011-06-26 23:18:58 +09:00
/* Memoy currently used by policy/audit log/query. */
unsigned int tomoyo_memory_used [ TOMOYO_MAX_MEMORY_STAT ] ;
/* Memory quota for "policy"/"audit log"/"query". */
unsigned int tomoyo_memory_quota [ TOMOYO_MAX_MEMORY_STAT ] ;
2010-05-17 10:12:46 +09:00
/**
* tomoyo_memory_ok - Check memory quota .
*
* @ ptr : Pointer to allocated memory .
*
* Returns true on success , false otherwise .
*
* Returns true if @ ptr is not NULL and quota not exceeded , false otherwise .
2011-09-25 17:51:06 +09:00
*
* Caller holds tomoyo_policy_lock mutex .
2010-05-17 10:12:46 +09:00
*/
bool tomoyo_memory_ok ( void * ptr )
{
2011-06-26 23:21:50 +09:00
if ( ptr ) {
const size_t s = ksize ( ptr ) ;
tomoyo_memory_used [ TOMOYO_MEMORY_POLICY ] + = s ;
2011-09-25 17:51:06 +09:00
if ( ! tomoyo_memory_quota [ TOMOYO_MEMORY_POLICY ] | |
tomoyo_memory_used [ TOMOYO_MEMORY_POLICY ] < =
tomoyo_memory_quota [ TOMOYO_MEMORY_POLICY ] )
2011-06-26 23:21:50 +09:00
return true ;
2011-09-25 17:51:06 +09:00
tomoyo_memory_used [ TOMOYO_MEMORY_POLICY ] - = s ;
2010-05-17 10:12:46 +09:00
}
tomoyo_warn_oom ( __func__ ) ;
return false ;
}
/**
* tomoyo_commit_ok - Check memory quota .
*
* @ data : Data to copy from .
* @ size : Size in byte .
*
* Returns pointer to allocated memory on success , NULL otherwise .
* @ data is zero - cleared on success .
2011-09-25 17:51:06 +09:00
*
* Caller holds tomoyo_policy_lock mutex .
2010-05-17 10:12:46 +09:00
*/
void * tomoyo_commit_ok ( void * data , const unsigned int size )
{
void * ptr = kzalloc ( size , GFP_NOFS ) ;
if ( tomoyo_memory_ok ( ptr ) ) {
memmove ( ptr , data , size ) ;
memset ( data , 0 , size ) ;
return ptr ;
}
2011-03-31 00:27:32 +09:00
kfree ( ptr ) ;
2010-05-17 10:12:46 +09:00
return NULL ;
}
2010-06-17 16:55:58 +09:00
/**
* tomoyo_get_group - Allocate memory for " struct tomoyo_path_group " / " struct tomoyo_number_group " .
*
2011-06-26 23:17:10 +09:00
* @ param : Pointer to " struct tomoyo_acl_param " .
* @ idx : Index number .
2010-06-17 16:55:58 +09:00
*
* Returns pointer to " struct tomoyo_group " on success , NULL otherwise .
*/
2011-06-26 23:17:10 +09:00
struct tomoyo_group * tomoyo_get_group ( struct tomoyo_acl_param * param ,
const u8 idx )
2010-06-17 16:55:58 +09:00
{
struct tomoyo_group e = { } ;
struct tomoyo_group * group = NULL ;
2011-06-26 23:17:10 +09:00
struct list_head * list ;
const char * group_name = tomoyo_read_token ( param ) ;
2010-06-17 16:55:58 +09:00
bool found = false ;
if ( ! tomoyo_correct_word ( group_name ) | | idx > = TOMOYO_MAX_GROUP )
return NULL ;
e . group_name = tomoyo_get_name ( group_name ) ;
if ( ! e . group_name )
return NULL ;
if ( mutex_lock_interruptible ( & tomoyo_policy_lock ) )
goto out ;
2011-06-26 23:19:52 +09:00
list = & param - > ns - > group_list [ idx ] ;
2011-06-26 23:17:10 +09:00
list_for_each_entry ( group , list , head . list ) {
2011-09-25 17:50:23 +09:00
if ( e . group_name ! = group - > group_name | |
atomic_read ( & group - > head . users ) = = TOMOYO_GC_IN_PROGRESS )
2010-06-17 16:55:58 +09:00
continue ;
2011-06-26 23:16:36 +09:00
atomic_inc ( & group - > head . users ) ;
2010-06-17 16:55:58 +09:00
found = true ;
break ;
}
if ( ! found ) {
struct tomoyo_group * entry = tomoyo_commit_ok ( & e , sizeof ( e ) ) ;
if ( entry ) {
INIT_LIST_HEAD ( & entry - > member_list ) ;
2011-06-26 23:16:36 +09:00
atomic_set ( & entry - > head . users , 1 ) ;
2011-06-26 23:17:10 +09:00
list_add_tail_rcu ( & entry - > head . list , list ) ;
2010-06-17 16:55:58 +09:00
group = entry ;
found = true ;
}
}
mutex_unlock ( & tomoyo_policy_lock ) ;
2011-06-26 23:17:10 +09:00
out :
2010-06-17 16:55:58 +09:00
tomoyo_put_name ( e . group_name ) ;
return found ? group : NULL ;
}
2010-05-17 10:12:46 +09:00
/*
* tomoyo_name_list is used for holding string data used by TOMOYO .
* Since same string data is likely used for multiple times ( e . g .
* " /lib/libc-2.5.so " ) , TOMOYO shares string data in the form of
* " const struct tomoyo_path_info * " .
*/
struct list_head tomoyo_name_list [ TOMOYO_MAX_HASH ] ;
/**
* tomoyo_get_name - Allocate permanent memory for string data .
*
* @ name : The string to store into the permernent memory .
*
* Returns pointer to " struct tomoyo_path_info " on success , NULL otherwise .
*/
const struct tomoyo_path_info * tomoyo_get_name ( const char * name )
{
2010-06-25 11:16:00 +09:00
struct tomoyo_name * ptr ;
2010-05-17 10:12:46 +09:00
unsigned int hash ;
int len ;
struct list_head * head ;
if ( ! name )
return NULL ;
len = strlen ( name ) + 1 ;
hash = full_name_hash ( ( const unsigned char * ) name , len - 1 ) ;
head = & tomoyo_name_list [ hash_long ( hash , TOMOYO_HASH_BITS ) ] ;
if ( mutex_lock_interruptible ( & tomoyo_policy_lock ) )
return NULL ;
2011-06-26 23:16:36 +09:00
list_for_each_entry ( ptr , head , head . list ) {
2011-09-25 17:50:23 +09:00
if ( hash ! = ptr - > entry . hash | | strcmp ( name , ptr - > entry . name ) | |
atomic_read ( & ptr - > head . users ) = = TOMOYO_GC_IN_PROGRESS )
2010-05-17 10:12:46 +09:00
continue ;
2011-06-26 23:16:36 +09:00
atomic_inc ( & ptr - > head . users ) ;
2010-05-17 10:12:46 +09:00
goto out ;
}
ptr = kzalloc ( sizeof ( * ptr ) + len , GFP_NOFS ) ;
2011-06-26 23:21:50 +09:00
if ( tomoyo_memory_ok ( ptr ) ) {
ptr - > entry . name = ( ( char * ) ptr ) + sizeof ( * ptr ) ;
memmove ( ( char * ) ptr - > entry . name , name , len ) ;
atomic_set ( & ptr - > head . users , 1 ) ;
tomoyo_fill_path_info ( & ptr - > entry ) ;
list_add_tail ( & ptr - > head . list , head ) ;
} else {
2010-05-17 10:12:46 +09:00
kfree ( ptr ) ;
ptr = NULL ;
}
2011-06-26 23:21:50 +09:00
out :
2010-05-17 10:12:46 +09:00
mutex_unlock ( & tomoyo_policy_lock ) ;
return ptr ? & ptr - > entry : NULL ;
}
2011-06-26 23:19:52 +09:00
/* Initial namespace.*/
struct tomoyo_policy_namespace tomoyo_kernel_namespace ;
2010-05-17 10:12:46 +09:00
/**
* tomoyo_mm_init - Initialize mm related code .
*/
void __init tomoyo_mm_init ( void )
{
int idx ;
for ( idx = 0 ; idx < TOMOYO_MAX_HASH ; idx + + )
INIT_LIST_HEAD ( & tomoyo_name_list [ idx ] ) ;
2011-06-26 23:19:52 +09:00
tomoyo_kernel_namespace . name = " <kernel> " ;
tomoyo_init_policy_namespace ( & tomoyo_kernel_namespace ) ;
tomoyo_kernel_domain . ns = & tomoyo_kernel_namespace ;
2010-05-17 10:12:46 +09:00
INIT_LIST_HEAD ( & tomoyo_kernel_domain . acl_info_list ) ;
2011-06-26 23:19:52 +09:00
tomoyo_kernel_domain . domainname = tomoyo_get_name ( " <kernel> " ) ;
2010-05-17 10:12:46 +09:00
list_add_tail_rcu ( & tomoyo_kernel_domain . list , & tomoyo_domain_list ) ;
}