You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

234 lines
7.3 KiB

/* BEGIN_LEGAL
Copyright (c) 2021 Intel Corporation
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
END_LEGAL */
/// @file xed-flags.h
///
#ifndef XED_FLAGS_H
# define XED_FLAGS_H
#include "xed-types.h"
#include "xed-portability.h"
#include "xed-flag-enum.h"
#include "xed-flag-action-enum.h"
#include "xed-gen-table-defs.h"
////////////////////////////////////////////////////////////////////////////
/// @ingroup FLAGS
/// a union of flags bits
union xed_flag_set_s {
xed_uint32_t flat;
struct {
xed_uint32_t cf:1; ///< bit 0
xed_uint32_t must_be_1:1;
xed_uint32_t pf:1;
xed_uint32_t must_be_0a:1;
xed_uint32_t af:1; ///< bit 4
xed_uint32_t must_be_0b:1;
xed_uint32_t zf:1;
xed_uint32_t sf:1;
xed_uint32_t tf:1; ///< bit 8
xed_uint32_t _if:1; ///< underscore to avoid token clash
xed_uint32_t df:1;
xed_uint32_t of:1;
xed_uint32_t iopl:2; ///< A 2-bit field, bits 12-13
xed_uint32_t nt:1;
xed_uint32_t must_be_0c:1;
xed_uint32_t rf:1; ///< bit 16
xed_uint32_t vm:1;
xed_uint32_t ac:1;
xed_uint32_t vif:1;
xed_uint32_t vip:1; ///< bit 20
xed_uint32_t id:1; ///< bit 21
xed_uint32_t must_be_0d:2; ///< bits 22-23
xed_uint32_t must_be_0e:4; ///< bits 24-27
// fc0,fc1,fc2,fc3 are not really part of rflags but I put them
// here to save space. These bits are only used for x87
// instructions.
xed_uint32_t fc0:1; ///< x87 flag FC0 (not really part of rflags)
xed_uint32_t fc1:1; ///< x87 flag FC1 (not really part of rflags)
xed_uint32_t fc2:1; ///< x87 flag FC2 (not really part of rflags)
xed_uint32_t fc3:1; ///< x87 flag FC3 (not really part of rflags)
} s;
};
typedef union xed_flag_set_s xed_flag_set_t;
/// @ingroup FLAGS
/// @name Flag-set accessors
//@{
/// @ingroup FLAGS
/// print the flag set in the supplied buffer
XED_DLL_EXPORT int xed_flag_set_print(const xed_flag_set_t* p, char* buf, int buflen);
/// @ingroup FLAGS
/// returns true if this object has a subset of the flags of the
/// "other" object.
XED_DLL_EXPORT xed_bool_t xed_flag_set_is_subset_of(const xed_flag_set_t* p,
const xed_flag_set_t* other);
//@}
////////////////////////////////////////////////////////////////////////////
/// @ingroup FLAGS
/// Associated with each flag field there can be one action.
typedef struct xed_flag_enum_s {
xed_flag_enum_t flag;
// there are at most two actions per flag. The 2nd may be invalid.
xed_flag_action_enum_t action;
} xed_flag_action_t;
/// @ingroup FLAGS
/// @name Lowest-level flag-action accessors
//@{
/// @ingroup FLAGS
/// get the name of the flag
XED_DLL_EXPORT xed_flag_enum_t
xed_flag_action_get_flag_name(const xed_flag_action_t* p);
/// @ingroup FLAGS
/// return the action
XED_DLL_EXPORT xed_flag_action_enum_t
xed_flag_action_get_action(const xed_flag_action_t* p, unsigned int i);
/// @ingroup FLAGS
/// returns true if the specified action is invalid. Only the 2nd flag might be invalid.
XED_DLL_EXPORT xed_bool_t
xed_flag_action_action_invalid(const xed_flag_action_enum_t a);
/// @ingroup FLAGS
/// print the flag & actions
XED_DLL_EXPORT int xed_flag_action_print(const xed_flag_action_t* p, char* buf, int buflen);
/// @ingroup FLAGS
/// returns true if either action is a read
XED_DLL_EXPORT xed_bool_t
xed_flag_action_read_flag(const xed_flag_action_t* p );
/// @ingroup FLAGS
/// returns true if either action is a write
XED_DLL_EXPORT xed_bool_t
xed_flag_action_writes_flag(const xed_flag_action_t* p);
/// @ingroup FLAGS
/// test to see if the specific action is a read
XED_DLL_EXPORT xed_bool_t
xed_flag_action_read_action( xed_flag_action_enum_t a);
/// @ingroup FLAGS
/// test to see if a specific action is a write
XED_DLL_EXPORT xed_bool_t
xed_flag_action_write_action( xed_flag_action_enum_t a);
//@}
////////////////////////////////////////////////////////////////////////////
/// @ingroup FLAGS
/// A collection of #xed_flag_action_t's and unions of read and written flags
typedef struct xed_simple_flag_s
{
///number of flag actions associated with this record
xed_uint8_t nflags;
xed_uint8_t may_write; /* 1/0, only using one bit */
xed_uint8_t must_write; /* 1/0, only using one bit */
///union of read flags
xed_flag_set_t read;
/// union of written flags (includes undefined flags);
xed_flag_set_t written;
/// union of undefined flags;
xed_flag_set_t undefined;
// index in to the xed_flag_action_table. nflags limits the # of entries.
xed_uint16_t fa_index;
} xed_simple_flag_t;
/// @ingroup FLAGS
/// @name Accessing the simple flags (Mid-level access)
//@{
/// @ingroup FLAGS
/// returns the number of flag-actions
XED_DLL_EXPORT unsigned int
xed_simple_flag_get_nflags(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// return union of bits for read flags
XED_DLL_EXPORT const xed_flag_set_t*
xed_simple_flag_get_read_flag_set(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// return union of bits for written flags
XED_DLL_EXPORT const xed_flag_set_t*
xed_simple_flag_get_written_flag_set(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// return union of bits for undefined flags
XED_DLL_EXPORT const xed_flag_set_t*
xed_simple_flag_get_undefined_flag_set(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// Indicates the flags are only conditionally written. Usually MAY-writes
/// of the flags instructions that are dependent on a REP count.
XED_DLL_EXPORT xed_bool_t xed_simple_flag_get_may_write(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// the flags always written
XED_DLL_EXPORT xed_bool_t xed_simple_flag_get_must_write(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// return the specific flag-action. Very detailed low level information
XED_DLL_EXPORT const xed_flag_action_t*
xed_simple_flag_get_flag_action(const xed_simple_flag_t* p, unsigned int i);
/// @ingroup FLAGS
/// boolean test to see if flags are read, scans the flags
XED_DLL_EXPORT xed_bool_t
xed_simple_flag_reads_flags(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// boolean test to see if flags are written, scans the flags
XED_DLL_EXPORT xed_bool_t xed_simple_flag_writes_flags(const xed_simple_flag_t* p);
/// @ingroup FLAGS
/// print the flags
XED_DLL_EXPORT int xed_simple_flag_print(const xed_simple_flag_t* p, char* buf, int buflen);
/// @ingroup FLAGS
/// Return the flags as a mask
static XED_INLINE unsigned int xed_flag_set_mask(const xed_flag_set_t* p) {
return p->flat; // FIXME: could mask out the X87 flags
}
//@}
////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////
#endif