2007-06-04 13:53:19 +04:00
/*
ctdb tunables code
Copyright ( C ) Andrew Tridgell 2007
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 09:29:31 +04:00
the Free Software Foundation ; either version 3 of the License , or
2007-06-04 13:53:19 +04:00
( at your option ) any later version .
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 .
You should have received a copy of the GNU General Public License
2007-07-10 09:29:31 +04:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2007-06-04 13:53:19 +04:00
*/
2015-10-26 08:50:46 +03:00
# include "replace.h"
# include "system/network.h"
# include <talloc.h>
# include <tdb.h>
# include "lib/util/debug.h"
# include "ctdb_private.h"
2007-06-04 13:53:19 +04:00
2015-10-23 06:17:34 +03:00
# include "common/common.h"
2015-11-11 07:41:10 +03:00
# include "common/logging.h"
2015-10-23 06:17:34 +03:00
2007-06-04 13:53:19 +04:00
static const struct {
const char * name ;
2007-06-04 14:22:44 +04:00
uint32_t default_v ;
2011-12-22 15:46:04 +04:00
size_t offset ;
bool obsolete ;
2007-06-04 13:53:19 +04:00
} tunable_map [ ] = {
2015-10-28 10:51:22 +03:00
{ " MaxRedirectCount " , 3 , offsetof ( struct ctdb_tunable_list , max_redirect_count ) , false } ,
{ " SeqnumInterval " , 1000 , offsetof ( struct ctdb_tunable_list , seqnum_interval ) , false } ,
{ " ControlTimeout " , 60 , offsetof ( struct ctdb_tunable_list , control_timeout ) , false } ,
{ " TraverseTimeout " , 20 , offsetof ( struct ctdb_tunable_list , traverse_timeout ) , false } ,
{ " KeepaliveInterval " , 5 , offsetof ( struct ctdb_tunable_list , keepalive_interval ) , false } ,
{ " KeepaliveLimit " , 5 , offsetof ( struct ctdb_tunable_list , keepalive_limit ) , false } ,
{ " RecoverTimeout " , 120 , offsetof ( struct ctdb_tunable_list , recover_timeout ) , false } ,
{ " RecoverInterval " , 1 , offsetof ( struct ctdb_tunable_list , recover_interval ) , false } ,
{ " ElectionTimeout " , 3 , offsetof ( struct ctdb_tunable_list , election_timeout ) , false } ,
{ " TakeoverTimeout " , 9 , offsetof ( struct ctdb_tunable_list , takeover_timeout ) , false } ,
{ " MonitorInterval " , 15 , offsetof ( struct ctdb_tunable_list , monitor_interval ) , false } ,
{ " TickleUpdateInterval " , 20 , offsetof ( struct ctdb_tunable_list , tickle_update_interval ) , false } ,
{ " EventScriptTimeout " , 30 , offsetof ( struct ctdb_tunable_list , script_timeout ) , false } ,
2015-10-28 09:03:01 +03:00
{ " MonitorTimeoutCount " , 20 , offsetof ( struct ctdb_tunable_list , monitor_timeout_count ) , false } ,
2015-10-28 10:51:22 +03:00
{ " EventScriptUnhealthyOnTimeout " , 0 , offsetof ( struct ctdb_tunable_list , script_unhealthy_on_timeout ) , true } ,
{ " RecoveryGracePeriod " , 120 , offsetof ( struct ctdb_tunable_list , recovery_grace_period ) , false } ,
{ " RecoveryBanPeriod " , 300 , offsetof ( struct ctdb_tunable_list , recovery_ban_period ) , false } ,
{ " DatabaseHashSize " , 100001 , offsetof ( struct ctdb_tunable_list , database_hash_size ) , false } ,
{ " DatabaseMaxDead " , 5 , offsetof ( struct ctdb_tunable_list , database_max_dead ) , false } ,
{ " RerecoveryTimeout " , 10 , offsetof ( struct ctdb_tunable_list , rerecovery_timeout ) , false } ,
{ " EnableBans " , 1 , offsetof ( struct ctdb_tunable_list , enable_bans ) , false } ,
{ " DeterministicIPs " , 0 , offsetof ( struct ctdb_tunable_list , deterministic_public_ips ) , false } ,
{ " LCP2PublicIPs " , 1 , offsetof ( struct ctdb_tunable_list , lcp2_public_ip_assignment ) , false } ,
{ " ReclockPingPeriod " , 60 , offsetof ( struct ctdb_tunable_list , reclock_ping_period ) , false } ,
{ " NoIPFailback " , 0 , offsetof ( struct ctdb_tunable_list , no_ip_failback ) , false } ,
{ " DisableIPFailover " , 0 , offsetof ( struct ctdb_tunable_list , disable_ip_failover ) , false } ,
{ " VerboseMemoryNames " , 0 , offsetof ( struct ctdb_tunable_list , verbose_memory_names ) , false } ,
{ " RecdPingTimeout " , 60 , offsetof ( struct ctdb_tunable_list , recd_ping_timeout ) , false } ,
{ " RecdFailCount " , 10 , offsetof ( struct ctdb_tunable_list , recd_ping_failcount ) , false } ,
{ " LogLatencyMs " , 0 , offsetof ( struct ctdb_tunable_list , log_latency_ms ) , false } ,
{ " RecLockLatencyMs " , 1000 , offsetof ( struct ctdb_tunable_list , reclock_latency_ms ) , false } ,
{ " RecoveryDropAllIPs " , 120 , offsetof ( struct ctdb_tunable_list , recovery_drop_all_ips ) , false } ,
{ " VerifyRecoveryLock " , 1 , offsetof ( struct ctdb_tunable_list , verify_recovery_lock ) , true } ,
{ " VacuumInterval " , 10 , offsetof ( struct ctdb_tunable_list , vacuum_interval ) , false } ,
{ " VacuumMaxRunTime " , 120 , offsetof ( struct ctdb_tunable_list , vacuum_max_run_time ) , false } ,
{ " RepackLimit " , 10000 , offsetof ( struct ctdb_tunable_list , repack_limit ) , false } ,
{ " VacuumLimit " , 5000 , offsetof ( struct ctdb_tunable_list , vacuum_limit ) , false } ,
{ " VacuumFastPathCount " , 60 , offsetof ( struct ctdb_tunable_list , vacuum_fast_path_count ) , false } ,
{ " MaxQueueDropMsg " , 1000000 , offsetof ( struct ctdb_tunable_list , max_queue_depth_drop_msg ) , false } ,
{ " AllowUnhealthyDBRead " , 0 , offsetof ( struct ctdb_tunable_list , allow_unhealthy_db_read ) , false } ,
{ " StatHistoryInterval " , 1 , offsetof ( struct ctdb_tunable_list , stat_history_interval ) , false } ,
{ " DeferredAttachTO " , 120 , offsetof ( struct ctdb_tunable_list , deferred_attach_timeout ) , false } ,
{ " AllowClientDBAttach " , 1 , offsetof ( struct ctdb_tunable_list , allow_client_db_attach ) , false } ,
{ " RecoverPDBBySeqNum " , 1 , offsetof ( struct ctdb_tunable_list , recover_pdb_by_seqnum ) , false } ,
{ " DeferredRebalanceOnNodeAdd " , 300 , offsetof ( struct ctdb_tunable_list , deferred_rebalance_on_node_add ) } ,
{ " FetchCollapse " , 1 , offsetof ( struct ctdb_tunable_list , fetch_collapse ) } ,
{ " HopcountMakeSticky " , 50 , offsetof ( struct ctdb_tunable_list , hopcount_make_sticky ) } ,
{ " StickyDuration " , 600 , offsetof ( struct ctdb_tunable_list , sticky_duration ) } ,
{ " StickyPindown " , 200 , offsetof ( struct ctdb_tunable_list , sticky_pindown ) } ,
{ " NoIPTakeover " , 0 , offsetof ( struct ctdb_tunable_list , no_ip_takeover ) , false } ,
{ " DBRecordCountWarn " , 100000 , offsetof ( struct ctdb_tunable_list , db_record_count_warn ) , false } ,
{ " DBRecordSizeWarn " , 10000000 , offsetof ( struct ctdb_tunable_list , db_record_size_warn ) , false } ,
{ " DBSizeWarn " , 100000000 , offsetof ( struct ctdb_tunable_list , db_size_warn ) , false } ,
{ " PullDBPreallocation " , 10 * 1024 * 1024 , offsetof ( struct ctdb_tunable_list , pulldb_preallocation_size ) , false } ,
{ " NoIPHostOnAllDisabled " , 0 , offsetof ( struct ctdb_tunable_list , no_ip_host_on_all_disabled ) , false } ,
{ " Samba3AvoidDeadlocks " , 0 , offsetof ( struct ctdb_tunable_list , samba3_hack ) , false } ,
{ " TDBMutexEnabled " , 0 , offsetof ( struct ctdb_tunable_list , mutex_enabled ) , false } ,
{ " LockProcessesPerDB " , 200 , offsetof ( struct ctdb_tunable_list , lock_processes_per_db ) , false } ,
2007-06-04 13:53:19 +04:00
} ;
2007-06-04 14:22:44 +04:00
/*
set all tunables to defaults
*/
void ctdb_tunables_set_defaults ( struct ctdb_context * ctdb )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( tunable_map ) ; i + + ) {
* ( uint32_t * ) ( tunable_map [ i ] . offset + ( uint8_t * ) & ctdb - > tunable ) = tunable_map [ i ] . default_v ;
}
}
2007-06-04 13:53:19 +04:00
/*
get a tunable
*/
int32_t ctdb_control_get_tunable ( struct ctdb_context * ctdb , TDB_DATA indata ,
TDB_DATA * outdata )
{
struct ctdb_control_get_tunable * t =
( struct ctdb_control_get_tunable * ) indata . dptr ;
char * name ;
uint32_t val ;
int i ;
if ( indata . dsize < sizeof ( * t ) | |
t - > length > indata . dsize - offsetof ( struct ctdb_control_get_tunable , name ) ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " Bad indata in ctdb_control_get_tunable \n " ) ) ;
2007-06-04 13:53:19 +04:00
return - 1 ;
}
name = talloc_strndup ( ctdb , ( char * ) t - > name , t - > length ) ;
CTDB_NO_MEMORY ( ctdb , name ) ;
for ( i = 0 ; i < ARRAY_SIZE ( tunable_map ) ; i + + ) {
if ( strcasecmp ( name , tunable_map [ i ] . name ) = = 0 ) break ;
}
talloc_free ( name ) ;
if ( i = = ARRAY_SIZE ( tunable_map ) ) {
2013-05-21 09:41:56 +04:00
return - EINVAL ;
2007-06-04 13:53:19 +04:00
}
val = * ( uint32_t * ) ( tunable_map [ i ] . offset + ( uint8_t * ) & ctdb - > tunable ) ;
outdata - > dptr = ( uint8_t * ) talloc ( outdata , uint32_t ) ;
CTDB_NO_MEMORY ( ctdb , outdata - > dptr ) ;
* ( uint32_t * ) outdata - > dptr = val ;
outdata - > dsize = sizeof ( uint32_t ) ;
return 0 ;
}
/*
set a tunable
*/
int32_t ctdb_control_set_tunable ( struct ctdb_context * ctdb , TDB_DATA indata )
{
2015-10-28 11:18:02 +03:00
struct ctdb_tunable_old * t =
( struct ctdb_tunable_old * ) indata . dptr ;
2007-06-04 13:53:19 +04:00
char * name ;
int i ;
if ( indata . dsize < sizeof ( * t ) | |
2015-10-28 11:18:02 +03:00
t - > length > indata . dsize - offsetof ( struct ctdb_tunable_old , name ) ) {
2008-02-04 12:07:15 +03:00
DEBUG ( DEBUG_ERR , ( " Bad indata in ctdb_control_set_tunable \n " ) ) ;
2007-06-04 13:53:19 +04:00
return - 1 ;
}
name = talloc_strndup ( ctdb , ( char * ) t - > name , t - > length ) ;
CTDB_NO_MEMORY ( ctdb , name ) ;
for ( i = 0 ; i < ARRAY_SIZE ( tunable_map ) ; i + + ) {
if ( strcasecmp ( name , tunable_map [ i ] . name ) = = 0 ) break ;
}
talloc_free ( name ) ;
2014-12-09 05:47:42 +03:00
2007-06-04 13:53:19 +04:00
if ( i = = ARRAY_SIZE ( tunable_map ) ) {
return - 1 ;
}
2009-06-25 08:45:17 +04:00
2007-06-04 13:53:19 +04:00
* ( uint32_t * ) ( tunable_map [ i ] . offset + ( uint8_t * ) & ctdb - > tunable ) = t - > value ;
2014-12-09 05:49:06 +03:00
if ( tunable_map [ i ] . obsolete ) {
DEBUG ( DEBUG_WARNING ,
( " Setting obsolete tunable \" %s \" \n " ,
tunable_map [ i ] . name ) ) ;
2015-02-09 02:32:47 +03:00
return 1 ;
2014-12-09 05:49:06 +03:00
}
2007-06-04 13:53:19 +04:00
return 0 ;
}
/*
list tunables
*/
int32_t ctdb_control_list_tunables ( struct ctdb_context * ctdb , TDB_DATA * outdata )
{
char * list = NULL ;
int i ;
struct ctdb_control_list_tunable * t ;
list = talloc_strdup ( outdata , tunable_map [ 0 ] . name ) ;
CTDB_NO_MEMORY ( ctdb , list ) ;
for ( i = 1 ; i < ARRAY_SIZE ( tunable_map ) ; i + + ) {
2011-12-22 15:52:55 +04:00
if ( tunable_map [ i ] . obsolete ) {
continue ;
}
2007-06-04 13:53:19 +04:00
list = talloc_asprintf_append ( list , " :%s " , tunable_map [ i ] . name ) ;
CTDB_NO_MEMORY ( ctdb , list ) ;
}
outdata - > dsize = offsetof ( struct ctdb_control_list_tunable , data ) +
strlen ( list ) + 1 ;
outdata - > dptr = talloc_size ( outdata , outdata - > dsize ) ;
CTDB_NO_MEMORY ( ctdb , outdata - > dptr ) ;
t = ( struct ctdb_control_list_tunable * ) outdata - > dptr ;
t - > length = strlen ( list ) + 1 ;
memcpy ( t - > data , list , t - > length ) ;
talloc_free ( list ) ;
return 0 ;
}