2003-07-04 22:34:56 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2003 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-20 20:55:30 +00:00
* Copyright ( C ) 2004 - 2006 Red Hat , Inc . All rights reserved .
2003-07-04 22:34:56 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-20 20:55:30 +00:00
* of the GNU Lesser General Public License v .2 .1 .
2004-03-30 19:35:44 +00:00
*
2007-08-20 20:55:30 +00:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 19:35:44 +00:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2003-07-04 22:34:56 +00:00
*/
# include "lib.h"
# include "memlock.h"
# include "defaults.h"
2004-03-26 21:11:34 +00:00
# include "config.h"
# include "toolcontext.h"
2003-07-04 22:34:56 +00:00
# include <limits.h>
# include <fcntl.h>
# include <unistd.h>
# include <sys/mman.h>
# include <sys/time.h>
# include <sys/resource.h>
# ifndef DEVMAPPER_SUPPORT
void memlock_inc ( void )
{
return ;
}
void memlock_dec ( void )
{
return ;
}
int memlock ( void )
{
return 0 ;
}
2003-08-20 15:48:27 +00:00
void memlock_init ( struct cmd_context * cmd )
{
return ;
}
2003-07-04 22:34:56 +00:00
# else /* DEVMAPPER_SUPPORT */
static size_t _size_stack ;
static size_t _size_malloc_tmp ;
static size_t _size_malloc = 2000000 ;
static void * _malloc_mem = NULL ;
static int _memlock_count = 0 ;
2009-11-19 01:11:57 +00:00
static int _memlock_count_daemon = 0 ;
2003-07-04 22:34:56 +00:00
static int _priority ;
static int _default_priority ;
static void _touch_memory ( void * mem , size_t size )
{
2006-08-17 18:23:44 +00:00
size_t pagesize = lvm_getpagesize ( ) ;
2003-07-04 22:34:56 +00:00
void * pos = mem ;
void * end = mem + size - sizeof ( long ) ;
while ( pos < end ) {
* ( long * ) pos = 1 ;
pos + = pagesize ;
}
}
static void _allocate_memory ( void )
{
void * stack_mem , * temp_malloc_mem ;
if ( ( stack_mem = alloca ( _size_stack ) ) )
_touch_memory ( stack_mem , _size_stack ) ;
if ( ( temp_malloc_mem = malloc ( _size_malloc_tmp ) ) )
_touch_memory ( temp_malloc_mem , _size_malloc_tmp ) ;
if ( ( _malloc_mem = malloc ( _size_malloc ) ) )
_touch_memory ( _malloc_mem , _size_malloc ) ;
free ( temp_malloc_mem ) ;
}
static void _release_memory ( void )
{
free ( _malloc_mem ) ;
}
/* Stop memory getting swapped out */
2006-04-19 15:33:07 +00:00
static void _lock_mem ( void )
2003-07-04 22:34:56 +00:00
{
# ifdef MCL_CURRENT
if ( mlockall ( MCL_CURRENT | MCL_FUTURE ) )
log_sys_error ( " mlockall " , " " ) ;
else
log_very_verbose ( " Locking memory " ) ;
# endif
_allocate_memory ( ) ;
errno = 0 ;
if ( ( ( _priority = getpriority ( PRIO_PROCESS , 0 ) ) = = - 1 ) & & errno )
log_sys_error ( " getpriority " , " " ) ;
else
if ( setpriority ( PRIO_PROCESS , 0 , _default_priority ) )
2008-05-28 23:12:45 +00:00
log_error ( " setpriority %d failed: %s " ,
2003-07-04 22:34:56 +00:00
_default_priority , strerror ( errno ) ) ;
}
2006-04-19 15:33:07 +00:00
static void _unlock_mem ( void )
2003-07-04 22:34:56 +00:00
{
# ifdef MCL_CURRENT
if ( munlockall ( ) )
log_sys_error ( " munlockall " , " " ) ;
else
log_very_verbose ( " Unlocking memory " ) ;
# endif
_release_memory ( ) ;
if ( setpriority ( PRIO_PROCESS , 0 , _priority ) )
log_error ( " setpriority %u failed: %s " , _priority ,
strerror ( errno ) ) ;
}
2009-11-19 01:11:57 +00:00
static void _lock_mem_if_needed ( void ) {
if ( ( _memlock_count + _memlock_count_daemon ) = = 1 )
_lock_mem ( ) ;
}
static void _unlock_mem_if_possible ( void ) {
if ( ( _memlock_count + _memlock_count_daemon ) = = 0 )
_unlock_mem ( ) ;
}
2003-07-04 22:34:56 +00:00
void memlock_inc ( void )
{
2009-11-19 01:11:57 +00:00
+ + _memlock_count ;
_lock_mem_if_needed ( ) ;
2003-07-04 22:34:56 +00:00
log_debug ( " memlock_count inc to %d " , _memlock_count ) ;
}
void memlock_dec ( void )
{
2009-11-19 01:11:57 +00:00
if ( ! _memlock_count )
2009-12-16 19:22:11 +00:00
log_error ( INTERNAL_ERROR " _memlock_count has dropped below 0. " ) ;
2009-11-19 01:11:57 +00:00
- - _memlock_count ;
_unlock_mem_if_possible ( ) ;
log_debug ( " memlock_count dec to %d " , _memlock_count ) ;
2003-07-04 22:34:56 +00:00
}
2009-11-19 01:11:57 +00:00
/*
* The memlock_ * _daemon functions will force the mlockall ( ) call that we need
* to stay in memory , but they will have no effect on device scans ( unlike
* normal memlock_inc and memlock_dec ) . Memory is kept locked as long as either
* of memlock or memlock_daemon is in effect .
*/
void memlock_inc_daemon ( void )
{
+ + _memlock_count_daemon ;
_lock_mem_if_needed ( ) ;
log_debug ( " memlock_count_daemon inc to %d " , _memlock_count_daemon ) ;
}
void memlock_dec_daemon ( void )
{
if ( ! _memlock_count_daemon )
2009-12-16 19:22:11 +00:00
log_error ( INTERNAL_ERROR " _memlock_count_daemon has dropped below 0. " ) ;
2009-11-19 01:11:57 +00:00
- - _memlock_count_daemon ;
_unlock_mem_if_possible ( ) ;
log_debug ( " memlock_count_daemon dec to %d " , _memlock_count_daemon ) ;
}
/*
* This disregards the daemon ( dmeventd ) locks , since we use memlock ( ) to check
* whether it is safe to run a device scan , which would normally coincide with
* ! memlock ( ) - - but the daemon global memory lock breaks this assumption , so
* we do not take those into account here .
*/
2003-07-04 22:34:56 +00:00
int memlock ( void )
{
return _memlock_count ;
}
void memlock_init ( struct cmd_context * cmd )
{
2006-05-16 16:48:31 +00:00
_size_stack = find_config_tree_int ( cmd ,
2003-07-04 22:34:56 +00:00
" activation/reserved_stack " ,
2004-03-08 18:28:45 +00:00
DEFAULT_RESERVED_STACK ) * 1024 ;
2006-05-16 16:48:31 +00:00
_size_malloc_tmp = find_config_tree_int ( cmd ,
2003-07-04 22:34:56 +00:00
" activation/reserved_memory " ,
2004-03-08 18:28:45 +00:00
DEFAULT_RESERVED_MEMORY ) * 1024 ;
2006-05-16 16:48:31 +00:00
_default_priority = find_config_tree_int ( cmd ,
2008-01-30 14:00:02 +00:00
" activation/process_priority " ,
DEFAULT_PROCESS_PRIORITY ) ;
2003-07-04 22:34:56 +00:00
}
# endif