blob: 1e09584db2ad646d8ff516ed22811805f6a67cdf [file] [log] [blame] [edit]
/* **********************************************************
* Copyright (c) 2012-2013 Google, Inc. All rights reserved.
* Copyright (c) 2003-2009 VMware, Inc. All rights reserved.
* **********************************************************/
/*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of VMware, Inc. nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL VMWARE, INC. OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*/
/* Copyright (c) 2003-2007 Determina Corp. */
/*
* options.h
*
* options struct and prototypes from options.c
* for use with core and gui/nm
*/
#ifndef _OPTIONS_H_
#define _OPTIONS_H_
#ifdef __cplusplus
extern "C"
{
#endif
/* Security policy flags. Note that if the off flag is not explicitly
* used in an option default value definition, then the option doesn't
* support that flag.
*/
typedef enum {
/* security mechanism needed for detection can be completely disabled */
OPTION_ENABLED = 0x1, /* if set, security mechanism is on */
OPTION_DISABLED = 0x0, /* if not set, security mechanism is off */
/* security policy can be built on top. For stateless policies
* all flags can be dynamic options. For stateful security mechanisms as
* long as OPTION_ENABLED stays on for gathering information, the
* security policies OPTION_BLOCK|OPTION_REPORT can be dynamically
* set. If no detection action is set, a detection check may be skipped.
*/
/* associated security policy can be either blocking or not */
/* If set, app action (i.e., bad behavior) is disallowed and remediation
* action (kill thread, kill process, throw exception) is performed.
* Note: detect_mode will override this flag.
* FIXME: app_thread_policy_helper seems to be the one place where
* detect_mode doesn't override this; case 9088 tracks this; xref case 8451
* for an explanation of why this seemed to leave as such.
*/
OPTION_BLOCK = 0x2,
OPTION_NO_BLOCK = 0x0,
/* policies that lend themselves to standard attack handling may use this flag */
OPTION_HANDLING = 0x4, /* if set, overrides default attack handling */
OPTION_NO_HANDLING = 0x0, /* when not set, default attack handling is used */
/* report or stay silent */
OPTION_REPORT = 0x8, /* if set, report action is being taken */
OPTION_NO_REPORT = 0x0, /* if not set, action is taken silently */
/* Block ignoring detect_mode; to handle case 10610. */
OPTION_BLOCK_IGNORE_DETECT = 0x20,
/* modifications in security policy or detection mechanism are controlled with */
OPTION_CUSTOM = 0x100, /* alternative policy bit - custom meaning per option */
OPTION_NO_CUSTOM = 0x0, /* alternative policy bit - custom meaning per option */
} security_option_t;
/* values taken by the option hook_conflict */
enum {
/* these are mutually exclusive */
HOOKED_TRAMPOLINE_DIE = 0, /* throw a fatal error if chained */
HOOKED_TRAMPOLINE_SQUASH = 1, /* modify any existing chains with a good guess */
HOOKED_TRAMPOLINE_CHAIN = 2, /* rerelativize and mangle to support chaining */
HOOKED_TRAMPOLINE_HOOK_DEEPER = 3, /* move our hook deeper into the function */
HOOKED_TRAMPOLINE_NO_HOOK = 4, /* give up on adding our hook */
HOOKED_TRAMPOLINE_MAX = 4,
};
/* for options.appfault_mask */
enum {
/* XXX: we don't raise on handled signals b/c nobody would want notification
* on timer signals. Should we raise on other handled signals?
*/
APPFAULT_FAULT = 0x0001, /* Unhandled signal, or any Windows exception */
APPFAULT_CRASH = 0x0002, /* Unhandled signal or exception (NYI on Windows) */
};
/* for all option uses */
/* N.B.: for 64-bit should we make the uint_size options be size_t?
* For now we're ok with all such options maxing out at 4GB, and in fact
* some like the fcache options are assumed to not be larger.
*/
#define uint_size uint
#define uint_time uint
/* So far all addr_t are external so we don't have a 64-bit problem */
#define uint_addr ptr_uint_t
/* XXX: For signed integer options, we'll need to correctly sign-extend in
* dr_get_integer_option.
*/
/* to dispatch on string default values, kept in struct not enum */
#define ISSTRING_bool 0
#define ISSTRING_uint 0
#define ISSTRING_uint_size 0
#define ISSTRING_uint_time 0
#define ISSTRING_ptr_uint_t 0
#define ISSTRING_pathstring_t 1
#define ISSTRING_liststring_t 1
/* Does this option affect persistent cache formation? */
typedef enum {
OP_PCACHE_NOP = 0, /* No effect on pcaches */
OP_PCACHE_LOCAL = 1, /* Can only relax (not tighten), and when it relaxes any
* module that module is marked via
* os_module_set_flag(MODULE_WAS_EXEMPTED).
*/
OP_PCACHE_GLOBAL = 2, /* Affects pcaches but not called out as local. */
} op_pcache_t;
/* We use an enum for default values of non-string options so that the compiler
* will fold them away (cl, at least, won't do it even for struct fields
* declared const), and also to determine whether options are internal or take
* string values or affect peristent caches.
* In order for this to work optionsx.h strings must either be EMPTY_STRING or
* a string in the OPTION_STRING() macro.
* N.B.: for 64-bit presumably every enum value is 64-bit-wide b/c some
* of them are. FIXME: could split up the enums to save space: or are we
* never actually storing these values?
*/
#define OPTION_STRING(x) 0 /* no string in enum */
#define EMPTY_STRING 0 /* no string in enum */
#define OPTION_COMMAND(type, name, default_value, command_line_option, statement, \
description, flag, pcache) \
OPTION_DEFAULT_VALUE_##name = (ptr_uint_t) default_value, \
OPTION_IS_INTERNAL_##name = false, \
OPTION_IS_STRING_##name = ISSTRING_##type, \
OPTION_AFFECTS_PCACHE_##name = pcache,
#define OPTION_COMMAND_INTERNAL(type, name, default_value, command_line_option, \
statement, description, flag, pcache) \
OPTION_DEFAULT_VALUE_##name = (ptr_uint_t) default_value, \
OPTION_IS_INTERNAL_##name = true, \
OPTION_IS_STRING_##name = ISSTRING_##type, \
OPTION_AFFECTS_PCACHE_##name = pcache,
enum option_is_internal {
# include "optionsx.h"
};
#undef OPTION_COMMAND
#undef OPTION_COMMAND_INTERNAL
#undef OPTION_STRING
#undef EMPTY_STRING
/* for all other option uses */
#define OPTION_STRING(x) x
#define EMPTY_STRING {0} /* fills the constant char structure with zeroes */
/* the Option struct typedef */
#ifdef EXPOSE_INTERNAL_OPTIONS
# define OPTION_COMMAND_INTERNAL(type, name, default_value, command_line_option, \
statement, description, flag, pcache) \
type name;
#else
# define OPTION_COMMAND_INTERNAL(type, name, default_value, command_line_option, \
statement, description, flag, pcache) /* nothing */
#endif
#define OPTION_COMMAND(type, name, default_value, command_line_option, statement, \
description, flag, pcache) \
type name;
typedef struct _options_t {
# include "optionsx.h"
} options_t;
#undef OPTION_COMMAND
#undef OPTION_COMMAND_INTERNAL
#ifndef EXPOSE_INTERNAL_OPTIONS
/* special struct for internal option default values */
# define OPTION_COMMAND(type, name, default_value, command_line_option, statement, \
description, flag, pcache) /* nothing */
# define OPTION_COMMAND_INTERNAL(type, name, default_value, command_line_option, \
statement, description, flag, pcache) \
const type name;
typedef struct _internal_options_t {
# include "optionsx.h"
} internal_options_t;
# undef OPTION_COMMAND
# undef OPTION_COMMAND_INTERNAL
#endif
#undef uint_size
#undef uint_time
#undef uint_addr
/* For default integer values we use an enum, while for default string values we
* use the default_options struct instance that is already being used
* for option parsing, and a second one for internal options when !INTERNAL
*/
extern const options_t default_options;
#ifndef EXPOSE_INTERNAL_OPTIONS
/* only needs to contain string options, but no compile-time way to
* do that without having OPTION_COMMAND_INTERNAL_STRING()!
*/
extern const internal_options_t default_internal_options;
#endif
#define IS_OPTION_INTERNAL(name) (OPTION_IS_INTERNAL_##name)
#define IS_OPTION_STRING(name) (OPTION_IS_STRING_##name)
/* FIXME : figure out a way to handle types here so that we don't have to cast
* strings to ints to avoid compiler warnings */
#define DEFAULT_OPTION_VALUE(name) (IS_OPTION_STRING(name) ? (int)default_options.name : \
OPTION_DEFAULT_VALUE_##name)
#ifdef EXPOSE_INTERNAL_OPTIONS
# define DEFAULT_INTERNAL_OPTION_VALUE DEFAULT_OPTION_VALUE
#else
# define DEFAULT_INTERNAL_OPTION_VALUE(name) (IS_OPTION_STRING(name) ? \
default_internal_options.name : \
OPTION_DEFAULT_VALUE_##name)
#endif
/* for non-internal builds we don't support setting the default value
* since they are all constants
*/
#define SET_DEFAULT_VALUE(name) (dynamo_options.name = DEFAULT_OPTION_VALUE(name))
/* checks for incompatible option values */
/* max==0 means no max and 0 is an ok value */
/* if option is incompatible, will try to touch up the option
* by assigning min to make it valid returns true if changed the
* option value
*/
bool
check_param_bounds(uint *val, uint min, uint max, const char *name);
int
options_init(void);
/* only frees a lock, does not destroy any options info other exit
* routines might need
*/
void
options_exit(void);
int
synchronize_dynamic_options(void);
#ifdef WINDOWS
const options_t *
get_process_options(HANDLE process_handle);
#endif
#define SYNCHRONIZE_DYNAMIC_OPTION(x) ( \
(dynamo_options.dynamic_options && synchronize_dynamic_options()), \
dynamo_options.x)
/*
* if minimal then the options string only contains values different than
* the defaults, otherwise it explicitly lists all options being used
*/
void
get_dynamo_options_string(options_t *options, char *opstr, int len, bool minimal);
/* Fills opstr with a minimal string of only
* persistent-cache-affecting options whose effect is >= pcache_effect
* and that are different from the defaults.
*/
void
get_pcache_dynamo_options_string(options_t *options, char *opstr, int len,
op_pcache_t pcache_effect);
bool
has_pcache_dynamo_options(options_t *options, op_pcache_t pcache_effect);
char *
parse_word(const char *str, const char **strpos, char *wordbuf, uint wordbuflen);
/****************************************************************************/
#ifdef NOT_DYNAMORIO_CORE
void
set_dynamo_options_defaults(options_t *options);
int
set_dynamo_options(options_t *options, const char *optstr);
#else /* !NOT_DYNAMORIO_CORE */
#include "utils.h"
/* are any fragments (potentially) shared? */
#define SHARED_FRAGMENTS_ENABLED() \
(DYNAMO_OPTION(shared_bbs) || DYNAMO_OPTION(shared_traces))
/* PR 244737: for x64 we use a "unified" scheme were thread-shared and
* thread-private code always uses TLS for scratch space, eliminating
* reachability issues. We require -private_ib_in_tls for x64 as well
* as for SHARED_FRAGMENTS_ENABLED and use that option to also
* cover non-ib scratch space.
*/
#define SCRATCH_ALWAYS_TLS() (DYNAMO_OPTION(private_ib_in_tls))
/* are any traces (potentially) private? */
/* FIXME Fix this if we permit private & shared traces to co-exist */
#define PRIVATE_TRACES_ENABLED() \
(!DYNAMO_OPTION(disable_traces) && !DYNAMO_OPTION(shared_traces))
/* are shared BBs ibl targets? */
#define SHARED_BB_IB_TARGETS() \
(DYNAMO_OPTION(shared_bbs) && DYNAMO_OPTION(bb_ibl_targets))
/* What this answers is
* "Are only shared BBs being created and are they valid IB targets?"
* It's used when a client wants to know if shared BBs are the only
* possible IB targets. So, for example, this is false whenever trace
* building is active, even if traces are not being added to the
* lookup tables.
*/
#define SHARED_BB_ONLY_IB_TARGETS() \
(SHARED_BB_IB_TARGETS() && DYNAMO_OPTION(disable_traces))
/* are any shared fragments ibl targets? */
#define SHARED_IB_TARGETS() \
(DYNAMO_OPTION(shared_traces) || SHARED_BB_IB_TARGETS())
/* are any IBT tables (potentially) shared? */
#define SHARED_IBT_TABLES_ENABLED() \
(DYNAMO_OPTION(shared_bb_ibt_tables) || DYNAMO_OPTION(shared_trace_ibt_tables))
#define TRACEDUMP_ENABLED() \
(!DYNAMO_OPTION(disable_traces) && \
(INTERNAL_OPTION(tracedump_text) || \
INTERNAL_OPTION(tracedump_binary) || \
INTERNAL_OPTION(tracedump_origins)))
#define RUNNING_WITHOUT_CODE_CACHE() \
(IF_HOTP(DYNAMO_OPTION(hotp_only) ||) DYNAMO_OPTION(thin_client))
#if defined(CLIENT_INTERFACE) && !defined(NOT_DYNAMORIO_CORE_PROPER)
# define CLIENT_OR_STANDALONE() \
(standalone_library || !IS_INTERNAL_STRING_OPTION_EMPTY(client_lib))
#else
# define CLIENT_OR_STANDALONE() false
#endif
extern char option_string[];
extern options_t dynamo_options;
extern read_write_lock_t options_lock;
/* check for empty is considered safe w/o the read lock
* this takes the field name only, and not {DYNAMO,INTERNAL}_OPTION macro,
* since those macros will ASSERT_OWN_READWRITE_LOCK(<is_stringtype>, &options_lock)
*/
#define IS_STRING_OPTION_EMPTY(op) ((dynamo_options.op)[0] == '\0')
/* single character check for ALL is considered safe w/o the read lock
* similarly to IS_STRING_OPTION_EMPTY see above
*/
#define IS_LISTSTRING_OPTION_FORALL(op) ((dynamo_options.op)[0] == '*')
#ifdef EXPOSE_INTERNAL_OPTIONS
# define IS_INTERNAL_STRING_OPTION_EMPTY(op) IS_STRING_OPTION_EMPTY(op)
#else
# define IS_INTERNAL_STRING_OPTION_EMPTY(op) ((default_internal_options.op)[0] == '\0')
#endif
/* 0=ret => 1, 1=call* => 2, 2=jmp* => 4 */
#define COARSE_FILL_IBL_MASK(branch_type) (1 << (branch_type))
/* full access to string requires read lock */
static inline void
string_option_read_lock() {
read_lock(&options_lock);
}
static inline void
string_option_read_unlock() {
read_unlock(&options_lock);
}
typedef enum {
LIST_NO_MATCH = 0, /* ensure can use as bool */
LIST_ON_DEFAULT = 1,
LIST_ON_APPEND = 2,
} list_default_or_append_t;
/* compare short_name, usually module name, against a list option of the combined
* default option (that could be overridden) and append list that is usually used
*/
list_default_or_append_t
check_list_default_and_append(liststring_t default_list, liststring_t append_list,
const char *short_name);
void
options_make_writable(void);
void
options_restore_readonly(void);
#endif /* !NOT_DYNAMORIO_CORE */
#ifdef __cplusplus
}
#endif
#endif /* _OPTIONS_H_ */