2013-11-25 03:13:10 +08:00
|
|
|
#ifndef NASM_IFLAG_H
|
|
|
|
#define NASM_IFLAG_H
|
|
|
|
|
|
|
|
#include "compiler.h"
|
2017-09-28 04:34:42 +08:00
|
|
|
#include "ilog2.h"
|
2013-11-25 03:13:10 +08:00
|
|
|
|
|
|
|
|
|
|
|
#include "iflaggen.h"
|
|
|
|
|
2019-08-07 13:18:34 +08:00
|
|
|
#define IF_GENBIT(bit) (UINT32_C(1) << ((bit) & 31))
|
|
|
|
|
|
|
|
static inline int ifcomp(uint32_t a, uint32_t b)
|
|
|
|
{
|
|
|
|
return (a > b) - (a < b);
|
|
|
|
}
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
static inline bool iflag_test(const iflag_t *f, unsigned int bit)
|
2013-11-25 03:13:10 +08:00
|
|
|
{
|
2019-08-07 13:18:34 +08:00
|
|
|
return !!(f->field[bit >> 5] & IF_GENBIT(bit));
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void iflag_set(iflag_t *f, unsigned int bit)
|
|
|
|
{
|
2019-08-07 13:18:34 +08:00
|
|
|
f->field[bit >> 5] |= IF_GENBIT(bit);
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void iflag_clear(iflag_t *f, unsigned int bit)
|
|
|
|
{
|
2019-08-07 13:18:34 +08:00
|
|
|
f->field[bit >> 5] &= ~IF_GENBIT(bit);
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void iflag_clear_all(iflag_t *f)
|
|
|
|
{
|
|
|
|
memset(f, 0, sizeof(*f));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void iflag_set_all(iflag_t *f)
|
|
|
|
{
|
2018-02-07 06:43:07 +08:00
|
|
|
memset(f, ~0, sizeof(*f));
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
#define iflag_for_each_field(v) for ((v) = 0; (v) < IF_FIELD_COUNT; (v)++)
|
|
|
|
|
2013-11-25 09:13:20 +08:00
|
|
|
static inline int iflag_cmp(const iflag_t *a, const iflag_t *b)
|
2013-11-25 03:13:10 +08:00
|
|
|
{
|
2013-11-27 07:21:15 +08:00
|
|
|
int i;
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
/* This is intentionally a reverse loop! */
|
|
|
|
for (i = IF_FIELD_COUNT-1; i >= 0; i--) {
|
2013-11-28 05:41:50 +08:00
|
|
|
if (a->field[i] == b->field[i])
|
|
|
|
continue;
|
|
|
|
|
2019-08-07 13:18:34 +08:00
|
|
|
return ifcomp(a->field[i], b->field[i]);
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define IF_GEN_HELPER(name, op) \
|
2013-11-25 09:13:20 +08:00
|
|
|
static inline iflag_t iflag_##name(const iflag_t *a, const iflag_t *b) \
|
2013-11-25 03:13:10 +08:00
|
|
|
{ \
|
|
|
|
unsigned int i; \
|
|
|
|
iflag_t res; \
|
|
|
|
\
|
2018-02-07 06:43:07 +08:00
|
|
|
iflag_for_each_field(i) \
|
2013-11-25 03:13:10 +08:00
|
|
|
res.field[i] = a->field[i] op b->field[i]; \
|
|
|
|
\
|
|
|
|
return res; \
|
|
|
|
}
|
|
|
|
|
|
|
|
IF_GEN_HELPER(xor, ^)
|
|
|
|
|
|
|
|
/* Some helpers which are to work with predefined masks */
|
2019-08-15 06:23:00 +08:00
|
|
|
#define IF_SMASK (IFM_SB|IFM_SW|IFM_SD|IFM_SQ|IFM_SO|IFM_SY|IFM_SZ|IFM_SIZE|IFM_ANYSIZE)
|
2019-08-07 13:18:34 +08:00
|
|
|
#define IF_ARMASK (IFM_AR0|IFM_AR1|IFM_AR2|IFM_AR3|IFM_AR4)
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2013-11-25 09:14:34 +08:00
|
|
|
#define _itemp_smask(idx) (insns_flags[(idx)].field[0] & IF_SMASK)
|
|
|
|
#define _itemp_armask(idx) (insns_flags[(idx)].field[0] & IF_ARMASK)
|
|
|
|
#define _itemp_arg(idx) ((_itemp_armask(idx) >> IF_AR0) - 1)
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2013-11-25 09:14:34 +08:00
|
|
|
#define itemp_smask(itemp) _itemp_smask((itemp)->iflag_idx)
|
|
|
|
#define itemp_arg(itemp) _itemp_arg((itemp)->iflag_idx)
|
|
|
|
#define itemp_armask(itemp) _itemp_armask((itemp)->iflag_idx)
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
/*
|
2019-08-07 13:05:49 +08:00
|
|
|
* IF_ANY is the highest CPU level by definition
|
2018-02-07 06:43:07 +08:00
|
|
|
*/
|
2019-08-07 13:05:49 +08:00
|
|
|
#define IF_PLEVEL IF_ANY /* Default CPU level */
|
2019-08-07 13:33:14 +08:00
|
|
|
#define IF_CPU_LEVEL_MASK ((IFM_ANY << 1) - 1)
|
2018-02-07 06:43:07 +08:00
|
|
|
|
|
|
|
static inline int iflag_cmp_cpu(const iflag_t *a, const iflag_t *b)
|
|
|
|
{
|
2019-08-07 13:18:34 +08:00
|
|
|
return ifcomp(a->field[IF_CPU_FIELD], b->field[IF_CPU_FIELD]);
|
2018-02-07 06:43:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t _iflag_cpu_level(const iflag_t *a)
|
|
|
|
{
|
|
|
|
return a->field[IF_CPU_FIELD] & IF_CPU_LEVEL_MASK;
|
|
|
|
}
|
|
|
|
|
2013-11-25 09:13:20 +08:00
|
|
|
static inline int iflag_cmp_cpu_level(const iflag_t *a, const iflag_t *b)
|
2013-11-25 03:13:10 +08:00
|
|
|
{
|
2019-08-07 13:18:34 +08:00
|
|
|
return ifcomp(_iflag_cpu_level(a), _iflag_cpu_level(b));
|
2018-02-07 06:43:07 +08:00
|
|
|
}
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
/* Returns true if the CPU level is at least a certain value */
|
|
|
|
static inline bool iflag_cpu_level_ok(const iflag_t *a, unsigned int bit)
|
|
|
|
{
|
2019-08-07 13:33:14 +08:00
|
|
|
return _iflag_cpu_level(a) >= IF_GENBIT(bit);
|
2018-02-07 06:43:07 +08:00
|
|
|
}
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
static inline void iflag_set_all_features(iflag_t *a)
|
|
|
|
{
|
2019-08-07 13:05:49 +08:00
|
|
|
uint32_t *p = &a->field[IF_FEATURE_FIELD];
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2019-08-07 13:05:49 +08:00
|
|
|
memset(p, -1, IF_FEATURE_NFIELDS * sizeof(uint32_t));
|
2018-02-07 06:43:07 +08:00
|
|
|
}
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
static inline void iflag_set_cpu(iflag_t *a, unsigned int cpu)
|
|
|
|
{
|
|
|
|
a->field[0] = 0; /* Not applicable to the CPU type */
|
|
|
|
iflag_set_all_features(a); /* All feature masking bits set for now */
|
|
|
|
a->field[IF_CPU_FIELD] &= ~IF_CPU_LEVEL_MASK;
|
|
|
|
iflag_set(a, cpu);
|
|
|
|
}
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2018-02-07 06:43:07 +08:00
|
|
|
static inline void iflag_set_default_cpu(iflag_t *a)
|
|
|
|
{
|
|
|
|
iflag_set_cpu(a, IF_PLEVEL);
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
2013-11-25 09:14:34 +08:00
|
|
|
static inline iflag_t _iflag_pfmask(const iflag_t *a)
|
2013-11-25 03:13:10 +08:00
|
|
|
{
|
2013-12-10 15:09:49 +08:00
|
|
|
iflag_t r;
|
|
|
|
|
|
|
|
iflag_clear_all(&r);
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2013-12-07 20:14:00 +08:00
|
|
|
if (iflag_test(a, IF_CYRIX))
|
|
|
|
iflag_set(&r, IF_CYRIX);
|
|
|
|
if (iflag_test(a, IF_AMD))
|
|
|
|
iflag_set(&r, IF_AMD);
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2013-12-07 20:14:00 +08:00
|
|
|
return r;
|
2013-11-25 03:13:10 +08:00
|
|
|
}
|
|
|
|
|
2013-12-07 20:14:00 +08:00
|
|
|
#define iflag_pfmask(itemp) _iflag_pfmask(&insns_flags[(itemp)->iflag_idx])
|
2013-11-25 03:13:10 +08:00
|
|
|
|
2013-11-25 09:14:34 +08:00
|
|
|
#endif /* NASM_IFLAG_H */
|