Initial commit: OpenRA game engine
Fork from OpenRA/OpenRA with one-click launch script (start-ra.cmd) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
329
OpenRA.Game/TraitDictionary.cs
Normal file
329
OpenRA.Game/TraitDictionary.cs
Normal file
@@ -0,0 +1,329 @@
|
||||
#region Copyright & License Information
|
||||
/*
|
||||
* Copyright (c) The OpenRA Developers and Contributors
|
||||
* This file is part of OpenRA, which is free software. It is made
|
||||
* available to you under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version. For more
|
||||
* information, see COPYING.
|
||||
*/
|
||||
#endregion
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Runtime.InteropServices;
|
||||
using OpenRA.Primitives;
|
||||
using OpenRA.Support;
|
||||
|
||||
namespace OpenRA
|
||||
{
|
||||
static class SpanExts
|
||||
{
|
||||
public static int BinarySearchMany(this Span<Actor> span, uint searchFor)
|
||||
{
|
||||
var start = 0;
|
||||
var end = span.Length;
|
||||
while (start != end)
|
||||
{
|
||||
var mid = (start + end) / 2;
|
||||
if (span[mid].ActorID < searchFor)
|
||||
start = mid + 1;
|
||||
else
|
||||
end = mid;
|
||||
}
|
||||
|
||||
return start;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Provides efficient ways to query a set of actors by their traits.
|
||||
/// </summary>
|
||||
sealed class TraitDictionary
|
||||
{
|
||||
static readonly Func<Type, ITraitContainer> CreateTraitContainer = t =>
|
||||
(ITraitContainer)typeof(TraitContainer<>).MakeGenericType(t).GetConstructor(Type.EmptyTypes).Invoke(null);
|
||||
|
||||
readonly Dictionary<Type, ITraitContainer> traits = [];
|
||||
|
||||
ITraitContainer InnerGet(Type t)
|
||||
{
|
||||
return traits.GetOrAdd(t, CreateTraitContainer);
|
||||
}
|
||||
|
||||
TraitContainer<T> InnerGet<T>()
|
||||
{
|
||||
return (TraitContainer<T>)InnerGet(typeof(T));
|
||||
}
|
||||
|
||||
public void PrintReport()
|
||||
{
|
||||
Log.AddChannel("traitreport", "traitreport.log");
|
||||
foreach (var t in traits.OrderByDescending(t => t.Value.Queries).TakeWhile(t => t.Value.Queries > 0))
|
||||
Log.Write("traitreport", $"{t.Key.Name}: {t.Value.Queries}");
|
||||
}
|
||||
|
||||
public void AddTrait(Actor actor, object val)
|
||||
{
|
||||
var t = val.GetType();
|
||||
|
||||
foreach (var i in t.GetInterfaces())
|
||||
InnerAdd(actor, i, val);
|
||||
foreach (var tt in t.BaseTypes())
|
||||
InnerAdd(actor, tt, val);
|
||||
}
|
||||
|
||||
void InnerAdd(Actor actor, Type t, object val)
|
||||
{
|
||||
InnerGet(t).Add(actor, val);
|
||||
}
|
||||
|
||||
static void CheckDestroyed(Actor actor)
|
||||
{
|
||||
if (actor.Disposed)
|
||||
throw new InvalidOperationException($"Attempted to get trait from destroyed object ({actor})");
|
||||
}
|
||||
|
||||
public T Get<T>(Actor actor)
|
||||
{
|
||||
CheckDestroyed(actor);
|
||||
return InnerGet<T>().Get(actor);
|
||||
}
|
||||
|
||||
public T GetOrDefault<T>(Actor actor)
|
||||
{
|
||||
CheckDestroyed(actor);
|
||||
return InnerGet<T>().GetOrDefault(actor);
|
||||
}
|
||||
|
||||
public IEnumerable<T> WithInterface<T>(Actor actor)
|
||||
{
|
||||
CheckDestroyed(actor);
|
||||
return InnerGet<T>().GetMultiple(actor.ActorID);
|
||||
}
|
||||
|
||||
public IEnumerable<TraitPair<T>> ActorsWithTrait<T>()
|
||||
{
|
||||
return InnerGet<T>().All();
|
||||
}
|
||||
|
||||
public IEnumerable<Actor> ActorsHavingTrait<T>()
|
||||
{
|
||||
return InnerGet<T>().Actors();
|
||||
}
|
||||
|
||||
public IEnumerable<Actor> ActorsHavingTrait<T>(Func<T, bool> predicate)
|
||||
{
|
||||
return InnerGet<T>().Actors(predicate);
|
||||
}
|
||||
|
||||
public void RemoveActor(Actor a)
|
||||
{
|
||||
foreach (var t in traits)
|
||||
t.Value.RemoveActor(a.ActorID);
|
||||
}
|
||||
|
||||
public void ApplyToActorsWithTrait<T>(Action<Actor, T> action)
|
||||
{
|
||||
InnerGet<T>().ApplyToAll(action);
|
||||
}
|
||||
|
||||
public void ApplyToActorsWithTraitTimed<T>(Action<Actor, T> action, string text)
|
||||
{
|
||||
InnerGet<T>().ApplyToAllTimed(action, text);
|
||||
}
|
||||
|
||||
interface ITraitContainer
|
||||
{
|
||||
void Add(Actor actor, object trait);
|
||||
void RemoveActor(uint actor);
|
||||
|
||||
int Queries { get; }
|
||||
}
|
||||
|
||||
sealed class TraitContainer<T> : ITraitContainer
|
||||
{
|
||||
readonly List<Actor> actors = [];
|
||||
readonly List<T> traits = [];
|
||||
|
||||
public int Queries { get; private set; }
|
||||
|
||||
public void Add(Actor actor, object trait)
|
||||
{
|
||||
var actorsSpan = CollectionsMarshal.AsSpan(actors);
|
||||
var insertIndex = actorsSpan.BinarySearchMany(actor.ActorID + 1);
|
||||
actors.Insert(insertIndex, actor);
|
||||
traits.Insert(insertIndex, (T)trait);
|
||||
}
|
||||
|
||||
public T Get(Actor actor)
|
||||
{
|
||||
var result = GetOrDefault(actor);
|
||||
if (result == null)
|
||||
throw new InvalidOperationException($"Actor {actor.Info.Name} does not have trait of type `{typeof(T)}`");
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public T GetOrDefault(Actor actor)
|
||||
{
|
||||
++Queries;
|
||||
var actorsSpan = CollectionsMarshal.AsSpan(actors);
|
||||
var index = actorsSpan.BinarySearchMany(actor.ActorID);
|
||||
if (index >= actorsSpan.Length || actorsSpan[index] != actor)
|
||||
return default;
|
||||
|
||||
if (index + 1 < actorsSpan.Length && actorsSpan[index + 1] == actor)
|
||||
throw new InvalidOperationException($"Actor {actor.Info.Name} has multiple traits of type `{typeof(T)}`");
|
||||
|
||||
return traits[index];
|
||||
}
|
||||
|
||||
public IEnumerable<T> GetMultiple(uint actor)
|
||||
{
|
||||
// PERF: Custom enumerator for efficiency - using `yield` is slower.
|
||||
++Queries;
|
||||
return new MultipleEnumerable(this, actor);
|
||||
}
|
||||
|
||||
sealed class MultipleEnumerable : IEnumerable<T>
|
||||
{
|
||||
readonly TraitContainer<T> container;
|
||||
readonly uint actor;
|
||||
public MultipleEnumerable(TraitContainer<T> container, uint actor) { this.container = container; this.actor = actor; }
|
||||
public IEnumerator<T> GetEnumerator() { return new MultipleEnumerator(container, actor); }
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { return GetEnumerator(); }
|
||||
}
|
||||
|
||||
struct MultipleEnumerator : IEnumerator<T>
|
||||
{
|
||||
readonly List<Actor> actors;
|
||||
readonly List<T> traits;
|
||||
readonly uint actor;
|
||||
int index;
|
||||
public MultipleEnumerator(TraitContainer<T> container, uint actor)
|
||||
: this()
|
||||
{
|
||||
actors = container.actors;
|
||||
traits = container.traits;
|
||||
this.actor = actor;
|
||||
Reset();
|
||||
}
|
||||
|
||||
public void Reset() { index = CollectionsMarshal.AsSpan(actors).BinarySearchMany(actor) - 1; }
|
||||
public bool MoveNext() { return ++index < actors.Count && actors[index].ActorID == actor; }
|
||||
public readonly T Current => traits[index];
|
||||
readonly object System.Collections.IEnumerator.Current => Current;
|
||||
public readonly void Dispose() { }
|
||||
}
|
||||
|
||||
public IEnumerable<TraitPair<T>> All()
|
||||
{
|
||||
// PERF: Custom enumerator for efficiency - using `yield` is slower.
|
||||
++Queries;
|
||||
return new AllEnumerable(this);
|
||||
}
|
||||
|
||||
public IEnumerable<Actor> Actors()
|
||||
{
|
||||
++Queries;
|
||||
Actor last = null;
|
||||
for (var i = 0; i < actors.Count; i++)
|
||||
{
|
||||
var current = actors[i];
|
||||
if (current == last)
|
||||
continue;
|
||||
|
||||
yield return current;
|
||||
last = current;
|
||||
}
|
||||
}
|
||||
|
||||
public IEnumerable<Actor> Actors(Func<T, bool> predicate)
|
||||
{
|
||||
++Queries;
|
||||
Actor last = null;
|
||||
|
||||
for (var i = 0; i < actors.Count; i++)
|
||||
{
|
||||
var current = actors[i];
|
||||
if (current == last || !predicate(traits[i]))
|
||||
continue;
|
||||
|
||||
yield return current;
|
||||
last = current;
|
||||
}
|
||||
}
|
||||
|
||||
readonly struct AllEnumerable : IEnumerable<TraitPair<T>>
|
||||
{
|
||||
readonly TraitContainer<T> container;
|
||||
public AllEnumerable(TraitContainer<T> container) { this.container = container; }
|
||||
public IEnumerator<TraitPair<T>> GetEnumerator() { return new AllEnumerator(container); }
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { return GetEnumerator(); }
|
||||
}
|
||||
|
||||
struct AllEnumerator : IEnumerator<TraitPair<T>>
|
||||
{
|
||||
readonly List<Actor> actors;
|
||||
readonly List<T> traits;
|
||||
int index;
|
||||
public AllEnumerator(TraitContainer<T> container)
|
||||
: this()
|
||||
{
|
||||
actors = container.actors;
|
||||
traits = container.traits;
|
||||
Reset();
|
||||
}
|
||||
|
||||
public void Reset() { index = -1; }
|
||||
public bool MoveNext() { return ++index < actors.Count; }
|
||||
public readonly TraitPair<T> Current => new(actors[index], traits[index]);
|
||||
readonly object System.Collections.IEnumerator.Current => Current;
|
||||
public readonly void Dispose() { }
|
||||
}
|
||||
|
||||
public void RemoveActor(uint actor)
|
||||
{
|
||||
var actorsSpan = CollectionsMarshal.AsSpan(actors);
|
||||
var startIndex = actorsSpan.BinarySearchMany(actor);
|
||||
if (startIndex >= actorsSpan.Length || actorsSpan[startIndex].ActorID != actor)
|
||||
return;
|
||||
|
||||
var endIndex = startIndex + 1;
|
||||
while (endIndex < actorsSpan.Length && actorsSpan[endIndex].ActorID == actor)
|
||||
endIndex++;
|
||||
|
||||
var count = endIndex - startIndex;
|
||||
actors.RemoveRange(startIndex, count);
|
||||
traits.RemoveRange(startIndex, count);
|
||||
}
|
||||
|
||||
public void ApplyToAll(Action<Actor, T> action)
|
||||
{
|
||||
var actorsSpan = CollectionsMarshal.AsSpan(actors);
|
||||
var traitsSpan = CollectionsMarshal.AsSpan(traits);
|
||||
|
||||
for (var i = 0; i < actorsSpan.Length; i++)
|
||||
action(actorsSpan[i], traitsSpan[i]);
|
||||
}
|
||||
|
||||
public void ApplyToAllTimed(Action<Actor, T> action, string text)
|
||||
{
|
||||
var start = PerfTickLogger.GetTimestamp();
|
||||
var actorsSpan = CollectionsMarshal.AsSpan(actors);
|
||||
var traitsSpan = CollectionsMarshal.AsSpan(traits);
|
||||
|
||||
for (var i = 0; i < actorsSpan.Length; i++)
|
||||
{
|
||||
var actor = actorsSpan[i];
|
||||
var trait = traitsSpan[i];
|
||||
action(actor, trait);
|
||||
|
||||
start = PerfTickLogger.LogLongTick(start, text, trait);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user