forked from simulationcraft/simc
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsc_death_knight.cpp
10813 lines (8863 loc) · 364 KB
/
sc_death_knight.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// ==========================================================================
// Dedmonwakeen's DPS-DPM Simulator.
// Send questions to [email protected]
// ==========================================================================
// TODO as of 2020-09-06
// Class:
// Killing Blow based mechanics (free Death Strike, Rune of Unending Thirst)
// Defensives: Anti-Magic Zone (+ group wide effect), Lichborne
// Disable noise from healing/defensive actions when simming a single, dps role, character
// Automate Rune energize in death_knight_action_t::execute() instead of per spell overrides
// utilize stat_pct_buffs instead of overriding player_t methods
// Add get_action() handling to diseases, obliterate/frost strike, active_spells.x
// Standardize debug_cast<T>() over other types of casting where possible
// Look into Death Strike OH handling (p -> dual_wield()?) and see if it can apply to other DW attacks
// Unholy:
// - Predict the first two Festering wounds on FS and use reaction time on the third?
// Blood:
// - Check that VB's absorb increase is correctly implemented
// - Healing from Consumption damage done
// Frost:
// - Revisit Deaths Due, Koltiras to verify 1H vs 2H behavior, especially with DD cleave
// - Test Frost's damage with atypical weapon setups (single 1H/OH, etc.)
// on abilities with the 2H penalty or combined AP type
// - Figure out what to do with Obliterate/Frost Strike strikes, reporting, etc.
// TOCHECK: accurate spawn/travel delay timers for all dynamic pets
// - DRW, Magus of the dead, Bloodworms have no delay at all
// - Army ghoul, Apoc ghoul, Raise dead ghoul have a different delay (currently estimated at 4.5s with a 0.1 stddev)
// TODO: even more variable travel time for apoc/army based on distance from boss on spawn?
// - Gargoyle has its own delay too
#include "simulationcraft.hpp"
#include "player/pet_spawner.hpp"
#include "action/action_callback.hpp"
#include "class_modules/apl/apl_death_knight.hpp"
namespace { // UNNAMED NAMESPACE
using namespace unique_gear;
struct death_knight_t;
struct runes_t;
struct rune_t;
namespace pets {
struct army_ghoul_pet_t;
struct bloodworm_pet_t;
struct dancing_rune_weapon_pet_t;
struct everlasting_bond_pet_t;
struct gargoyle_pet_t;
struct ghoul_pet_t;
struct magus_pet_t;
}
namespace runeforge {
void apocalypse( special_effect_t& );
void fallen_crusader( special_effect_t& );
void hysteria( special_effect_t& );
void razorice( special_effect_t& );
void sanguination( special_effect_t& );
void spellwarding( special_effect_t& );
void stoneskin_gargoyle( special_effect_t& );
void unending_thirst( special_effect_t& ); // Effect only procs on killing blows, NYI
}
enum runeforge_apocalypse { DEATH, FAMINE, PESTILENCE, WAR, MAX };
// ==========================================================================
// Death Knight Runes ( part 1 )
// ==========================================================================
enum rune_state { STATE_DEPLETED, STATE_REGENERATING, STATE_FULL };
const double RUNIC_POWER_DECAY_RATE = 1.0;
const double RUNE_REGEN_BASE = 10;
const double RUNE_REGEN_BASE_SEC = ( 1 / RUNE_REGEN_BASE );
const size_t MAX_RUNES = 6;
const size_t MAX_REGENERATING_RUNES = 3;
const double MAX_START_OF_COMBAT_RP = 20;
template <typename T>
struct dynamic_event_t : public event_t
{
double m_coefficient;
T** m_ptr;
dynamic_event_t( sim_t* s ) : event_t( *s ), m_coefficient( 1.0 ), m_ptr( nullptr )
{ }
const char* name() const override
{ return "Dynamic-Event-Base"; }
static T* create( sim_t* s )
{ return new (*s) T( s ); }
virtual T* clone() = 0;
virtual timespan_t adjust( timespan_t by_time )
{
auto this_ = this;
// Execute early and cancel the event, if the adjustment would trigger the event
if ( by_time >= remains() )
{
execute();
event_t::cancel( this_ );
return 0_ms;
}
auto new_remains = remains() - by_time;
sim().print_debug( "{} adjust time by {}, remains= {}, new_remains={}",
name(), by_time.total_seconds(), remains().total_seconds(), new_remains.total_seconds() );
// Otherwise, just clone this event and schedule it with the new time, bypassing the coefficient
// adjustment
create_clone() -> schedule( new_remains, false );
event_t::cancel( this_ );
return new_remains;
}
// Create a clone of this event
virtual T* create_clone()
{
auto cloned = clone();
if ( m_ptr )
{
cloned -> ptr( *m_ptr );
*m_ptr = cloned;
}
cloned -> coefficient( m_coefficient );
return cloned;
}
virtual void execute_event() = 0;
// Update the duration coefficient, causing an adjustment to the remaining event duration to be
// made
virtual T* update_coefficient( double new_coefficient )
{
if ( new_coefficient == 0 || new_coefficient == m_coefficient )
{
return cast();
}
auto ratio = new_coefficient / m_coefficient;
auto remains = this -> remains();
auto new_duration = remains * ratio;
sim().print_debug( "{} coefficient change, remains={} old_coeff={} new_coeff={} ratio={} new_remains={}",
name(), remains.total_seconds(), m_coefficient, new_coefficient, ratio, new_duration.total_seconds() );
// Duration increases, so reschedule the event
if ( ratio > 1 )
{
reschedule( new_duration );
m_coefficient = new_coefficient;
return cast();
}
// Duration decreases, cannot reschedule so clone the event and schedule it with the new
// remaining duration, bypassing the coefficient adjustment (since it is already included in the
// duration)
else
{
auto cloned = create_clone();
cloned -> coefficient( new_coefficient )
-> schedule( new_duration, false );
// Cancel wants ref to a ptr, but we don't really care
auto this_ = this;
event_t::cancel( this_ );
return cast( cloned );
}
}
static T* cast( event_t* event )
{ return static_cast<T*>( event ); }
T* cast() const
{ return static_cast<T*>( this ); }
T* cast()
{ return static_cast<T*>( this ); }
T* ptr( T*& value )
{ m_ptr = &value; return cast(); }
T* ptr() const
{ return m_ptr ? *m_ptr : nullptr; }
T* coefficient( double value )
{ m_coefficient = value; return cast(); }
double coefficient() const
{ return m_coefficient; }
// Actually schedules the event into the core event system, by default, applies the coefficient
// associated with the event to the duration given
T* schedule( timespan_t duration, bool use_coeff = true )
{
event_t::schedule( duration * ( use_coeff ? coefficient() : 1.0 ) );
return cast();
}
void execute() override
{
execute_event();
if ( m_ptr )
{
*m_ptr = nullptr;
}
}
};
struct rune_event_t : public dynamic_event_t<rune_event_t>
{
using super = dynamic_event_t<rune_event_t>;
rune_t* m_rune;
rune_event_t( sim_t* sim ) :
super( sim ), m_rune( nullptr )
{ }
const char* name() const override
{ return "Rune-Regen-Event"; }
rune_event_t* clone() override
{ return create( &sim() ) -> rune( *m_rune ); }
rune_t* rune() const
{ return m_rune; }
rune_event_t* rune( rune_t& r )
{ m_rune = &r; return this; }
void execute_event() override;
};
struct rune_t
{
runes_t* runes; // Back reference to runes_t array so we can adjust rune state
rune_state state; // DEPLETED, REGENERATING, FULL
rune_event_t* event; // Regen event
timespan_t regen_start; // Start time of the regeneration
timespan_t regenerated; // Timestamp when rune regenerated to full
rune_t() : runes( nullptr ), state( STATE_FULL ), event( nullptr ), regen_start( timespan_t::min() )
{ }
rune_t( runes_t* r ) : runes( r ), state( STATE_FULL ), event( nullptr ), regen_start( timespan_t::min() )
{ }
bool is_ready() const { return state == STATE_FULL ; }
bool is_regenerating() const { return state == STATE_REGENERATING; }
bool is_depleted() const { return state == STATE_DEPLETED; }
double fill_level() const;
void update_coefficient();
// Consume this rune and adjust internal rune state
void consume();
// Fill this rune and adjust internal rune state
void fill_rune( gain_t* gain = nullptr );
// Start regenerating the rune
void start_regenerating();
// Directly adjust regeneration by time units
void adjust_regen_event( timespan_t adjustment );
void reset()
{
regen_start = timespan_t::min();
regenerated = 0_ms;
event = nullptr;
state = STATE_FULL;
}
};
struct runes_t
{
death_knight_t* dk;
std::array<rune_t, MAX_RUNES> slot;
timespan_t waste_start;
// Cumulative waste per iteration in seconds
extended_sample_data_t cumulative_waste;
// Individual waste times per rune in seconds
extended_sample_data_t rune_waste;
// Per iteration waste counter, added into cumulative_waste on reset
timespan_t iteration_waste_sum;
// Cached regenerating runes array
std::vector<const rune_t*> regenerating_runes;
runes_t( death_knight_t* p );
void update_coefficient()
{ range::for_each( slot, []( rune_t& r ) { r.update_coefficient(); } ); }
void reset()
{
range::for_each( slot, []( rune_t& r ) { r.reset(); } );
waste_start = 0_ms;
if ( iteration_waste_sum > 0_ms )
{
cumulative_waste.add( iteration_waste_sum.total_seconds() );
}
iteration_waste_sum = 0_ms;
}
std::string string_representation() const
{
std::string rune_str;
std::string rune_val_str;
for ( const auto& rune: slot )
{
char rune_letter;
if ( rune.is_ready() ) {
rune_letter = 'F';
} else if ( rune.is_depleted() ) {
rune_letter = 'd';
} else {
rune_letter = 'r';
}
std::string rune_val = util::to_string( rune.fill_level(), 2 );
rune_str += rune_letter;
rune_val_str += '[' + rune_val + ']';
}
return rune_str + " " + rune_val_str;
}
// Return the number of runes in specific state
unsigned runes_in_state( rune_state s ) const
{
return std::accumulate( slot.begin(), slot.end(), 0U,
[ s ]( const unsigned& v, const rune_t& r ) { return v + ( r.state == s ); });
}
// Return the first rune in a specific state. If no rune in specific state found, return nullptr.
rune_t* first_rune_in_state( rune_state s )
{
auto it = range::find_if( slot, [ s ]( const rune_t& rune ) { return rune.state == s; } );
if ( it != slot.end() )
{
return &(*it);
}
return nullptr;
}
unsigned runes_regenerating() const
{ return runes_in_state( STATE_REGENERATING ); }
unsigned runes_depleted() const
{ return runes_in_state( STATE_DEPLETED ); }
unsigned runes_full() const
{ return runes_in_state( STATE_FULL ); }
rune_t* first_depleted_rune()
{ return first_rune_in_state( STATE_DEPLETED ); }
rune_t* first_regenerating_rune()
{ return first_rune_in_state( STATE_REGENERATING ); }
rune_t* first_full_rune()
{ return first_rune_in_state( STATE_FULL ); }
void consume( unsigned runes );
// Perform seconds of rune regeneration time instantaneously
void regenerate_immediate( timespan_t seconds );
// Time it takes with the current rune regeneration speed to regenerate n_runes by the Death
// Knight.
timespan_t time_to_regen( unsigned n_runes );
};
// Finds an action with the given name. If no action exists, a new one will
// be created.
//
// Use this with secondary background actions to ensure the player only has
// one copy of the action.
// Shamelessly borrowed from the mage module
template <typename Action, typename Actor, typename... Args>
action_t* get_action( util::string_view name, Actor* actor, Args&&... args )
{
action_t* a = actor->find_action( name );
if ( !a )
a = new Action( name, actor, std::forward<Args>( args )... );
assert( dynamic_cast<Action*>( a ) && a->name_str == name && a->background );
return a;
}
// ==========================================================================
// Death Knight
// ==========================================================================
struct death_knight_td_t : public actor_target_data_t {
struct
{
// Blood
dot_t* blood_plague;
// Frost
dot_t* frost_fever;
// Unholy
dot_t* soul_reaper;
dot_t* virulent_plague;
} dot;
struct
{
// Runeforges
buff_t* apocalypse_death; // Dummy debuff, healing reduction not implemented
buff_t* apocalypse_war;
buff_t* apocalypse_famine;
buff_t* razorice;
// General Talents
buff_t* brittle;
// Blood
buff_t* mark_of_blood;
buff_t* tightening_grasp;
// Frost
buff_t* piercing_chill;
buff_t* everfrost;
// Unholy
buff_t* festering_wound;
buff_t* unholy_blight;
buff_t* rotten_touch;
buff_t* death_rot;
} debuff;
death_knight_td_t( player_t* target, death_knight_t* p );
};
using data_t = std::pair<std::string, simple_sample_data_with_min_max_t>;
using simple_data_t = std::pair<std::string, simple_sample_data_t>;
struct death_knight_t : public player_t {
public:
// Stores the currently active death and decay ground event
ground_aoe_event_t* active_dnd;
// Expression warnings
// for old dot.death_and_decay.x expressions
bool deprecated_dnd_expression;
// for runeforge.name expressions that call death knight runeforges instead of shadowlands legendary runeforges
bool runeforge_expression_warning;
// Counters
unsigned int km_proc_attempts; // critical auto attacks since the last KM proc
unsigned int festering_wounds_target_count; // cached value of the current number of enemies affected by FW
unsigned int bone_shield_charges_consumed; // Counts how many bone shield charges have been consumed for T29 4pc blood
stats_t* antimagic_shell;
stats_t* antimagic_zone;
// Data collection for cooldown waste
auto_dispose<std::vector<data_t*>> cd_waste_exec, cd_waste_cumulative;
auto_dispose<std::vector<simple_data_t*>> cd_waste_iter;
// Buffs
struct buffs_t {
// Shared
buff_t* antimagic_shell;
buff_t* antimagic_zone;
buff_t* icebound_fortitude;
buff_t* rune_mastery;
buff_t* unholy_ground;
buff_t* icy_talons;
buff_t* empower_rune_weapon;
buff_t* abomination_limb;
buff_t* lichborne; // NYI
// Runeforges
buff_t* rune_of_hysteria;
buff_t* stoneskin_gargoyle;
buff_t* unholy_strength; // runeforge of the fallen crusader
// Blood
absorb_buff_t* blood_shield;
buff_t* bone_shield;
buff_t* coagulopathy;
buff_t* crimson_scourge;
buff_t* dancing_rune_weapon;
buff_t* heartrend;
buff_t* hemostasis;
buff_t* ossuary;
buff_t* perseverance_of_the_ebon_blade;
buff_t* rune_tap;
buff_t* sanguine_ground;
buff_t* tombstone;
buff_t* vampiric_blood;
buff_t* vigorous_lifeblood_4pc; // T29 4pc
buff_t* voracious;
// Frost
buff_t* breath_of_sindragosa;
buff_t* cold_heart;
buff_t* gathering_storm;
buff_t* inexorable_assault;
buff_t* killing_machine;
buff_t* pillar_of_frost;
buff_t* pillar_of_frost_bonus; // Additional strength from runes spent
buff_t* remorseless_winter;
buff_t* rime;
buff_t* unleashed_frenzy;
buff_t* bonegrinder_crit;
buff_t* bonegrinder_frost;
buff_t* enduring_strength_builder;
buff_t* enduring_strength;
buff_t* frostwhelps_aid;
buff_t* shattering_blade;
// Unholy
buff_t* dark_transformation;
buff_t* runic_corruption;
buff_t* sudden_doom;
buff_t* unholy_assault;
buff_t* unholy_blight;
buff_t* unholy_pact;
buff_t* festermight;
buff_t* ghoulish_frenzy;
buff_t* plaguebringer;
buff_t* ghoulish_infusion;
buff_t* commander_of_the_dead_window;
} buffs;
struct runeforge_t {
bool rune_of_apocalypse;
bool rune_of_hysteria;
// Razorice has one for each weapon because they don't proc from the same abilities
bool rune_of_razorice_mh, rune_of_razorice_oh;
bool rune_of_sanguination;
bool rune_of_the_fallen_crusader;
bool rune_of_the_stoneskin_gargoyle;
bool rune_of_unending_thirst;
// Spellwarding is simpler to store as a double to check whether it's stacked or not
double rune_of_spellwarding;
} runeforge;
// Cooldowns
struct cooldowns_t {
// Shared
cooldown_t* abomination_limb;
cooldown_t* death_and_decay_dynamic; // Shared cooldown object for death and decay, defile and death's due
cooldown_t* mind_freeze;
// Blood
cooldown_t* bone_shield_icd; // internal cooldown between bone shield stack consumption
cooldown_t* blood_tap;
cooldown_t* dancing_rune_weapon;
cooldown_t* vampiric_blood;
// Frost
cooldown_t* icecap_icd; // internal cooldown that prevents several procs on the same dual-wield attack
cooldown_t* inexorable_assault_icd; // internal cooldown to prevent multiple procs during aoe
cooldown_t* koltiras_favor_icd; // internal cooldown that prevents several procs on the same dual-wield attack
cooldown_t* frigid_executioner_icd; // internal cooldown that prevents several procs on the same dual-wield attack
cooldown_t* enduring_strength_icd; // internal cooldown that prevents several procs on the same dual-wield attacl
cooldown_t* pillar_of_frost;
// Unholy
cooldown_t* apocalypse;
cooldown_t* army_of_the_dead;
cooldown_t* dark_transformation;
cooldown_t* vile_contagion;
} cooldown;
// Active Spells
struct active_spells_t {
// Shared
action_t* runeforge_pestilence;
action_t* runeforge_razorice;
action_t* runeforge_sanguination;
// Class Tree
action_t* blood_draw;
// Blood
action_t* mark_of_blood_heal;
action_t* shattering_bone;
// Unholy
action_t* bursting_sores;
action_t* festering_wound;
action_t* virulent_eruption;
action_t* ruptured_viscera;
} active_spells;
// Gains
struct gains_t {
// Shared
gain_t* antimagic_shell; // RP from magic damage absorbed
gain_t* antimagic_zone; // RP from magic damage absorbed
gain_t* rune; // Rune regeneration
gain_t* rune_of_hysteria;
gain_t* spirit_drain;
gain_t* coldthirst;
gain_t* start_of_combat_overflow;
// Blood
gain_t* blood_tap;
gain_t* bryndaors_might;
gain_t* drw_heart_strike; // Blood Strike, Blizzard's hack to replicate HS rank 2 with DRW
gain_t* heartbreaker;
gain_t* tombstone;
gain_t* vigorous_lifeblood_2pc; // T29 2pc
// Frost
gain_t* breath_of_sindragosa;
gain_t* empower_rune_weapon;
gain_t* frost_fever; // RP generation per tick
gain_t* horn_of_winter;
gain_t* koltiras_favor;
gain_t* frigid_executioner; // Rune refund chance
gain_t* murderous_efficiency;
gain_t* obliteration;
gain_t* rage_of_the_frozen_champion;
gain_t* runic_attenuation;
gain_t* runic_empowerment;
// Unholy
gain_t* apocalypse;
gain_t* festering_wound;
gain_t* replenishing_wounds;
gain_t* feasting_strikes;
} gains;
// Specialization
struct specialization_t {
// Class/spec auras
const spell_data_t* death_knight;
const spell_data_t* plate_specialization;
const spell_data_t* riposte;
const spell_data_t* blood_fortification;
const spell_data_t* blood_death_knight;
const spell_data_t* frost_death_knight;
const spell_data_t* unholy_death_knight;
// Shared
const spell_data_t* frost_fever; // The RP energize spell is a spec ability in spelldata
const spell_data_t* death_coil;
const spell_data_t* death_grip;
const spell_data_t* dark_command;
const spell_data_t* death_and_decay;
const spell_data_t* rune_strike;
// Unholy
const spell_data_t* dark_transformation_2;
const spell_data_t* festering_wound;
} spec;
// Mastery
struct mastery_t {
const spell_data_t* blood_shield; // Blood
const spell_data_t* frozen_heart; // Frost
const spell_data_t* dreadblade; // Unholy
} mastery;
// Talents
struct talents_t {
// Shared Class Tree
player_talent_t chains_of_ice;
player_talent_t death_strike;
player_talent_t raise_dead;
// Row 2
player_talent_t mind_freeze;
player_talent_t antimagic_shell;
player_talent_t cleaving_strikes;
// Row 3
player_talent_t blinding_sleet; // NYI
player_talent_t coldthirst;
player_talent_t permafrost; // NYI
player_talent_t improved_death_strike;
player_talent_t antimagic_barrier;
player_talent_t march_of_darkness; // NYI
player_talent_t sacrificial_pact;
player_talent_t control_undead; // NYI
player_talent_t enfeeble; // NYI
// Row 4
player_talent_t icebound_fortitude;
player_talent_t blood_scent;
player_talent_t veteran_of_the_third_war;
player_talent_t suppression; // NYI
player_talent_t brittle;
// Row 5
player_talent_t acclimation; // NYI
player_talent_t merciless_strikes;
player_talent_t antimagic_zone; // NYI
player_talent_t might_of_thassarian;
player_talent_t clenching_grasp; // NYI
// Row 6
player_talent_t proliferating_chill;
player_talent_t asphyxiate; // NYI
player_talent_t assimilation; // NYI
player_talent_t death_pact; // NYI
player_talent_t grip_of_the_dead; // NYI
player_talent_t deaths_reach; // NYI
player_talent_t unholy_endurance; // NYI
player_talent_t gloom_ward; // NYI
// Row 7
player_talent_t runic_attenuation;
player_talent_t wraith_walk; // NYI
player_talent_t unholy_ground;
player_talent_t insidious_chill; // NYI
// Row 8
player_talent_t blood_draw;
player_talent_t will_of_the_necropolis; // NYI
player_talent_t deaths_echo;
// Row 9
player_talent_t icy_talons;
player_talent_t rune_mastery;
player_talent_t unholy_bond;
// Row 10
player_talent_t empower_rune_weapon;
player_talent_t abomination_limb;
player_talent_t soul_reaper;
// Blood
struct {
// Row 1
player_talent_t heart_strike;
// Row 2
player_talent_t marrowrend;
player_talent_t blood_boil;
// Row 3
player_talent_t vampiric_blood;
player_talent_t crimson_scourge;
// Row 4
player_talent_t ossuary;
player_talent_t improved_vampiric_blood;
player_talent_t improved_heart_strike;
player_talent_t deaths_caress;
// Row 5
player_talent_t rune_tap;
player_talent_t heartbreaker;
player_talent_t foul_bulwark;
player_talent_t dancing_rune_weapon;
player_talent_t hemostasis;
player_talent_t perseverance_of_the_ebon_blade;
player_talent_t relish_in_blood;
// Row 6
player_talent_t leeching_strike;
player_talent_t improved_bone_shield;
player_talent_t insatiable_blade;
player_talent_t blooddrinker;
player_talent_t consumption;
player_talent_t rapid_decomposition;
// Row 7
player_talent_t blood_feast;
player_talent_t blood_tap;
player_talent_t reinforced_bones;
player_talent_t everlasting_bond;
player_talent_t voracious;
player_talent_t bloodworms;
player_talent_t coagulopathy;
// Row 8
player_talent_t mark_of_blood;
player_talent_t tombstone;
player_talent_t gorefiends_grasp;
player_talent_t sanguine_ground;
// Row 9
player_talent_t shattering_bone;
player_talent_t heartrend;
player_talent_t tightening_grasp;
player_talent_t iron_heart;
player_talent_t red_thirst;
// Row 10
player_talent_t bonestorm;
player_talent_t purgatory;
player_talent_t bloodshot;
player_talent_t umbilicus_eternus;
} blood;
// Frost
struct {
// Row 1
player_talent_t frost_strike;
// Row 2
player_talent_t obliterate;
player_talent_t howling_blast;
// Row 3
player_talent_t killing_machine;
player_talent_t rime;
// Row 4
player_talent_t unleashed_frenzy;
player_talent_t runic_command;
player_talent_t improved_frost_strike;
player_talent_t remorseless_winter;
// Row 5
player_talent_t improved_obliterate;
player_talent_t glacial_advance;
player_talent_t pillar_of_frost;
player_talent_t improved_rime;
player_talent_t frostscythe;
// Row 6
player_talent_t rage_of_the_frozen_champion;
player_talent_t frigid_executioner;
player_talent_t horn_of_winter;
player_talent_t frostreaper;
player_talent_t enduring_strength;
player_talent_t frostwhelps_aid;
player_talent_t cold_heart;
player_talent_t biting_cold;
player_talent_t chill_streak;
// Row 7
player_talent_t murderous_efficiency;
player_talent_t inexorable_assault;
player_talent_t icecap;
player_talent_t empower_rune_weapon;
player_talent_t gathering_storm;
player_talent_t enduring_chill;
player_talent_t piercing_chill;
// Row 8
player_talent_t might_of_the_frozen_wastes;
player_talent_t bonegrinder;
player_talent_t shattering_blade;
player_talent_t avalanche;
player_talent_t icebreaker;
player_talent_t everfrost;
// Row 9
player_talent_t cold_blooded_rage;
player_talent_t frostwyrms_fury;
player_talent_t invigorating_freeze;
// Row 10
player_talent_t obliteration;
player_talent_t absolute_zero;
player_talent_t breath_of_sindragosa;
} frost;
// Unholy
struct {
// Row 1
player_talent_t festering_strike;
// Row 2
player_talent_t scourge_strike;
player_talent_t raise_dead;
// Row 3
player_talent_t outbreak;
player_talent_t dark_transformation;
// Row 4
player_talent_t unholy_blight;
player_talent_t improved_festering_strike; // Horrible naming convention, plz change blizz
player_talent_t runic_mastery;
player_talent_t infected_claws;
// Row 5
player_talent_t epidemic;
player_talent_t replenishing_wounds;
player_talent_t feasting_strikes;
player_talent_t apocalypse;
player_talent_t clawing_shadows;
player_talent_t plaguebringer;
player_talent_t sudden_doom;
player_talent_t all_will_serve;
// Row 6
player_talent_t pestilent_pustules;
player_talent_t bursting_sores;
player_talent_t ebon_fever;
player_talent_t unholy_command;
player_talent_t magus_of_the_dead;
player_talent_t ruptured_viscera;
player_talent_t improved_death_coil;
player_talent_t rotten_touch;
player_talent_t unholy_pact;
player_talent_t defile;
// Row 7
player_talent_t vile_contagion;
player_talent_t pestilence;
player_talent_t eternal_agony;
player_talent_t coil_of_devastation;
player_talent_t harbinger_of_doom;
player_talent_t reaping;
// Row 8
player_talent_t death_rot;
player_talent_t army_of_the_dead;
player_talent_t summon_gargoyle;
// Row 9
player_talent_t festermight;
player_talent_t ghoulish_frenzy;
player_talent_t army_of_the_damned;
player_talent_t morbidity;
player_talent_t unholy_aura;
// Row 10
player_talent_t unholy_assault;
player_talent_t superstrain;
player_talent_t commander_of_the_dead;
} unholy;
} talent;
// Spells
struct spells_t {
// Shared
const spell_data_t* brittle_debuff;
const spell_data_t* dnd_buff; // obliterate aoe increase while in death's due (nf covenant ability)
const spell_data_t* razorice_debuff;
const spell_data_t* rune_mastery_buff;
const spell_data_t* empower_rune_weapon_main; // Empower Rune Weapon has a unique ID for the spell itself, with each talent just modifying number of charges.
const spell_data_t* coldthirst_gain; // Coldthirst has a unique ID for the gain and cooldown reduction
// Diseases (because they're not stored in spec data, unlike frost fever's rp gen...)
const spell_data_t* blood_plague;
const spell_data_t* frost_fever;
const spell_data_t* virulent_plague;
// Blood
const spell_data_t* blood_shield;
const spell_data_t* bone_shield;
const spell_data_t* sanguine_ground;
const spell_data_t* tightening_grasp_debuff;
// Frost
const spell_data_t* runic_empowerment_gain;
const spell_data_t* murderous_efficiency_gain;
const spell_data_t* rage_of_the_frozen_champion; // RP generation spell
const spell_data_t* piercing_chill_debuff;
const spell_data_t* runic_empowerment_chance;
// Unholy
const spell_data_t* runic_corruption; // buff
const spell_data_t* runic_corruption_chance;
const spell_data_t* festering_wound_debuff;
const spell_data_t* rotten_touch_debuff;
const spell_data_t* death_rot_debuff;
const spell_data_t* coil_of_devastation_debuff;
const spell_data_t* feasting_strikes_gain;
const spell_data_t* ghoulish_frenzy_player;
const spell_data_t* plaguebringer_buff;
const spell_data_t* festermight_buff;
const spell_data_t* ghoulish_infusion;
const spell_data_t* unholy_blight_dot;
// T29 Blood
const spell_data_t* vigorous_lifeblood_4pc; // Damage and haste buff
const spell_data_t* vigorous_lifeblood_energize; // Rune refund
} spell;
// Unholy Pet Abilities
struct pet_spells_t {
// Raise dead ghoul
const spell_data_t* ghoul_claw;
const spell_data_t* sweeping_claws;
const spell_data_t* gnaw;
const spell_data_t* monstrous_blow;
// Army of the dead
const spell_data_t* army_claw;
// Gargoyle
const spell_data_t* gargoyle_strike;
const spell_data_t* dark_empowerment;
// All Will Serve skulker
const spell_data_t* skulker_shot;
// Army of the damned magus
const spell_data_t* frostbolt;
const spell_data_t* shadow_bolt;
// Commander of the Dead Talent
const spell_data_t* commander_of_the_dead;
// Ruptured Viscera Talent
const spell_data_t* ruptured_viscera;
// Ghoulish Frenzy
const spell_data_t* ghoulish_frenzy;
// Vile Infusion
const spell_data_t* vile_infusion;
} pet_spell;
// RPPM
struct rppm_t
{
real_ppm_t* bloodworms;
real_ppm_t* runic_attenuation;
} rppm;
// Pets and Guardians
struct pets_t
{
pets::dancing_rune_weapon_pet_t* dancing_rune_weapon_pet;
pets::dancing_rune_weapon_pet_t* everlasting_bond_pet;
pets::gargoyle_pet_t* gargoyle;
pets::ghoul_pet_t* ghoul_pet;
pet_t* risen_skulker;
spawner::pet_spawner_t<pets::army_ghoul_pet_t, death_knight_t> army_ghouls;
spawner::pet_spawner_t<pets::army_ghoul_pet_t, death_knight_t> apoc_ghouls;
spawner::pet_spawner_t<pets::bloodworm_pet_t, death_knight_t> bloodworms;