From f9d37c36363be935ee5cdf50fcc6f747fde6c3e9 Mon Sep 17 00:00:00 2001 From: DavidJCobb <831497+DavidJCobb@users.noreply.github.com> Date: Wed, 16 Oct 2024 22:49:53 +0200 Subject: [PATCH 1/2] Update Actor.h: parent class offsets Parent classes are listed with incorrect offsets; did TESObjectREFR grow larger by 8 bytes at some point? Verified through automated examination of the RTTI's complete object locator (I have a tool that can pretty-print them), and through manual examination of the Actor constructor to see where the VTBLs are being written. --- include/RE/A/Actor.h | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/include/RE/A/Actor.h b/include/RE/A/Actor.h index f64741a4b..8269915fa 100644 --- a/include/RE/A/Actor.h +++ b/include/RE/A/Actor.h @@ -124,12 +124,12 @@ namespace RE class Actor : public TESObjectREFR, // 000 - public MagicTarget, // 098 - public ActorValueOwner, // 0B0 - public ActorState, // 0B8 - public BSTEventSink, // 0C8 - public BSTEventSink, // 0D0 - public IPostAnimationChannelUpdateFunctor // 0D8 + public MagicTarget, // 0A0 + public ActorValueOwner, // 0B8 + public ActorState, // 0C0 + public BSTEventSink, // 0D0 + public BSTEventSink, // 0D8 + public IPostAnimationChannelUpdateFunctor // 0E0 { private: using EntryPoint = BGSEntryPointPerkEntry::EntryPoint; From fbbeebd7b9a5e50ed0d02a0b16ac060f5b62aedc Mon Sep 17 00:00:00 2001 From: DavidJCobb <831497+DavidJCobb@users.noreply.github.com> Date: Wed, 16 Oct 2024 23:29:10 +0200 Subject: [PATCH 2/2] Update ActorValueInfo.h All of my ActorValueInfo research to date. Most of it is based in LE, but I double-checked in SSE to verify the offsets of type, computeBaseFunc, enumValueCount, enumValues, and isNthAICached[Whatever]Value. --- include/RE/A/ActorValueInfo.h | 148 +++++++++++++++++++++++++--------- 1 file changed, 110 insertions(+), 38 deletions(-) diff --git a/include/RE/A/ActorValueInfo.h b/include/RE/A/ActorValueInfo.h index bd4f8ef41..0d18a522a 100644 --- a/include/RE/A/ActorValueInfo.h +++ b/include/RE/A/ActorValueInfo.h @@ -9,6 +9,7 @@ namespace RE { + class ActorValueOwner; class BGSSkillPerkTreeNode; class ActorValueInfo : @@ -21,6 +22,8 @@ namespace RE inline static constexpr auto RTTI = RTTI_ActorValueInfo; inline static constexpr auto VTABLE = VTABLE_ActorValueInfo; inline static constexpr auto FORMTYPE = FormType::ActorValueInfo; + + using DynamicBaseValueFunctor = float (*)(ActorValueOwner*, uint32_t avIndex); struct RecordFlags { @@ -31,6 +34,84 @@ namespace RE }; }; + enum class ActorValueType : std::uint32_t + { + kAttribute, // e.g. Health, HealRate + kSkill, // e.g. OneHanded, VampirePerks, WerewolfPerks + kAITemperament, // e.g. Aggression, Energy, Morality + kDamageResistance, // e.g. PoisonResist + kLimbCondition, // e.g. BrainCondition + kStatus, // e.g. DetectLifeRange, Invisibility, NightEye, Paralysis + kMiscellaneous, // e.g. ArmorPerks, BowSpeedBonus, Fame, [Skill]Mod, [Skill]SkillAdvance, Variable01, WardPower + }; + + enum class ActorValueFlag : std::uint32_t + { + // used for combat-related magic skills + kHostileEffectsScaleWithDifficulty = 1 << 1, + + // affects kClampAsSpecialStat behavior + kSpecialStatClampsAsNonZero = 1 << 2, + + // clamped value (ActorValueOwner::GetClampedActorValue) is in the range [0, 10] + // or [1, 10] + kClampAsSpecialStat = 1 << 3, + + // clamped value is in the range [0, 100] + kClampAsSkill = 1 << 4, + + // the value can have "permanent," "temporary," and "damage" modifiers + // (see RE::Modifiers, RE::ActorValueStorage, RE::Actor::healthModifiers, etc.) + kCanHaveModifiers = 1 << 5, + + // if the base value is computed from the actor or race, then the computed + // base value is the dynamic computation + the current value + kBaseValueIsDynamicPlusCurrent = 1 << 6, + + // base value is computed per reference (`computeBaseFunc`), and the GetAVInfo + // console command lists the "derived" value as "ignored" + kBaseValueComputedFromActor = 1 << 7, + + // the AV is an enum. + kEnumeration = 1 << 8, + + // damaging this AV increases the value, and restoring it decreases the value. + // used for AVs where high = bad, e.g. MovementNoiseMult impairing sneak + kInverted = 1 << 9, + + // base value is computed per reference (`computeBaseFunc`), and values cached + // in the AI process (see other flags) are forcibly refreshed on race change or + // savegame load + kBaseValueComputedFromRace = 1 << 11, + + // Papyrus APIs refuse to alter this AV's value + kCannotBeAlteredByScripts = 1 << 14, + + kBaseValueIsAlwaysZero = 1 << 15, + + // probably intended for multipliers, though some (e.g. WeaponSpeedMult) use the + // kBaseValueIsAlwaysZero flag instead + kBaseValueIsAlwaysOne = 1 << 16, + + // used for limb condition AVs + kBaseValueIsAlwaysOneHundred = 1 << 17, + + // see RE::CachedValues::actorValueCache + kAIProcessCachesCurrentValue = 1 << 18, + + // see RE::CachedValues::maxActorValueCache. non-numeric AVs like Confidence, in + // particular, only cache their current value and not a maximum + kAIProcessCachesMaxValue = 1 << 19, + + // the current value for the player cannot be decreased while ToggleGodMode + // is active + kProtectedByGodMode = 1 << 20, + + // when an effect's token is substituted for the magnitude, the displayed + // value is scaled by 100 + kDisplayedEffectMagnitudeTimesOneHundred = 1 << 21, + }; + struct Skill // AVSK { float useMult; // 00 @@ -51,44 +132,35 @@ namespace RE [[nodiscard]] const char* GetDefaultPath() const override; // 06 - { return "Textures\\"; } // members - const char* enumName; // 050 - BSFixedString abbreviation; // 058 - ANAM - std::uint32_t flags; // 060 - std::uint32_t unk064; // 064 - std::uint64_t unk068; // 068 - std::uint32_t unk070; // 070 - std::uint32_t unk074; // 074 - std::uint32_t unk078; // 078 - std::uint32_t unk07C; // 07C - std::uint32_t unk080; // 080 - std::uint32_t unk084; // 084 - std::uint32_t unk088; // 088 - std::uint32_t unk08C; // 08C - std::uint32_t unk090; // 090 - std::uint32_t unk094; // 094 - std::uint32_t unk098; // 098 - std::uint32_t unk09C; // 09C - std::uint32_t unk0A0; // 0A0 - std::uint32_t unk0A4; // 0A4 - std::uint32_t unk0A8; // 0A8 - std::uint32_t unk0AC; // 0AC - std::uint64_t unk0B0; // 0B0 - std::uint64_t unk0B8; // 0B8 - std::uint64_t unk0C0; // 0C0 - std::uint64_t unk0C8; // 0C8 - std::uint64_t unk0D0; // 0D0 - std::uint64_t unk0D8; // 0D8 - std::uint64_t unk0E0; // 0E0 - std::uint64_t unk0E8; // 0E8 - std::uint64_t unk0F0; // 0F0 - std::uint64_t unk0F8; // 0F8 - std::uint64_t unk100; // 100 - Skill* skill; // 108 - AVSK - std::uint32_t unk110; // 110 - std::uint32_t unk114; // 114 - BGSSkillPerkTreeNode* perkTree; // 118 - std::uint32_t perkTreeWidth; // 120 - std::uint32_t unk124; // 124 - CNAM + const char* enumName; // 050 + BSFixedString abbreviation; // 058 - ANAM + REX::EnumSet flags; // 060 + ActorValueType type; // 064 + DynamicBaseValueFunctor computeBaseFunc; // 068 - used if appropriate flags are set + std::uint32_t unk070; // 070 + std::uint32_t unk074; // 074 + std::uint32_t unk078; // 078 + std::uint32_t unk07C; // 07C + std::uint32_t unk080; // 080 + std::uint32_t unk084; // 084 + std::uint32_t unk088; // 088 + std::uint32_t unk08C; // 08C + std::uint32_t unk090; // 090 + std::uint32_t unk094; // 094 + std::uint32_t unk098; // 098 + std::uint32_t unk09C; // 09C + std::uint32_t unk0A0; // 0A0 + std::uint32_t unk0A4; // 0A4 + std::uint32_t unk0A8; // 0A8 + std::uint32_t unk0AC; // 0AC + std::uint64_t enumValueCount; // 0B0 + const char* enumValues[0xA]; // 0B8 + Skill* skill; // 108 - AVSK + std::uint32_t isNthAICachedCurrentValue; // 110 - if appropriate AV flag is set, RE::CachedValues::actorValueCache[n] is this AV + std::uint32_t isNthAICachedMaxValue; // 114 - if appropriate AV flag is set, RE::CachedValues::maxActorValueCache[n] is this AV + BGSSkillPerkTreeNode* perkTree; // 118 + std::uint32_t perkTreeWidth; // 120 + std::uint32_t unk124; // 124 - CNAM }; static_assert(sizeof(ActorValueInfo) == 0x128); }