mirror of
https://github.com/ppy/osu.git
synced 2025-01-21 11:22:55 +08:00
Fix SegmentedGraph
draw node calculating segment colours in unsafe manner
The `SegmentedGraph`'s draw node would call `getSegmentColour()` on the drawable, which would query the `DrawColourInfo` and `tierColours` properties of the drawable. This is a cross-thread access and as such completely unsafe, as due to being cross-thread it can die on invalidations or out-of-bounds accesses. Fix by transferring everything to the draw node first before attempting to draw. `SegmentedGraph.TierColours` setter already correctly invalidates the draw node via `graphNeedsUpdate`, so no further intervention was required there. Closes #22326.
This commit is contained in:
parent
292fd345bc
commit
718cbf9382
@ -152,22 +152,6 @@ namespace osu.Game.Graphics.UserInterface
|
|||||||
segments.Sort();
|
segments.Sort();
|
||||||
}
|
}
|
||||||
|
|
||||||
private ColourInfo getSegmentColour(SegmentInfo segment)
|
|
||||||
{
|
|
||||||
var segmentColour = new ColourInfo
|
|
||||||
{
|
|
||||||
TopLeft = DrawColourInfo.Colour.Interpolate(new Vector2(segment.Start, 0f)),
|
|
||||||
TopRight = DrawColourInfo.Colour.Interpolate(new Vector2(segment.End, 0f)),
|
|
||||||
BottomLeft = DrawColourInfo.Colour.Interpolate(new Vector2(segment.Start, 1f)),
|
|
||||||
BottomRight = DrawColourInfo.Colour.Interpolate(new Vector2(segment.End, 1f))
|
|
||||||
};
|
|
||||||
|
|
||||||
var tierColour = segment.Tier >= 0 ? tierColours[segment.Tier] : new Colour4(0, 0, 0, 0);
|
|
||||||
segmentColour.ApplyChild(tierColour);
|
|
||||||
|
|
||||||
return segmentColour;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override DrawNode CreateDrawNode() => new SegmentedGraphDrawNode(this);
|
protected override DrawNode CreateDrawNode() => new SegmentedGraphDrawNode(this);
|
||||||
|
|
||||||
protected struct SegmentInfo
|
protected struct SegmentInfo
|
||||||
@ -215,6 +199,7 @@ namespace osu.Game.Graphics.UserInterface
|
|||||||
private IShader shader = null!;
|
private IShader shader = null!;
|
||||||
private readonly List<SegmentInfo> segments = new List<SegmentInfo>();
|
private readonly List<SegmentInfo> segments = new List<SegmentInfo>();
|
||||||
private Vector2 drawSize;
|
private Vector2 drawSize;
|
||||||
|
private readonly List<Colour4> tierColours = new List<Colour4>();
|
||||||
|
|
||||||
public SegmentedGraphDrawNode(SegmentedGraph<T> source)
|
public SegmentedGraphDrawNode(SegmentedGraph<T> source)
|
||||||
: base(source)
|
: base(source)
|
||||||
@ -228,8 +213,12 @@ namespace osu.Game.Graphics.UserInterface
|
|||||||
texture = Source.texture;
|
texture = Source.texture;
|
||||||
shader = Source.shader;
|
shader = Source.shader;
|
||||||
drawSize = Source.DrawSize;
|
drawSize = Source.DrawSize;
|
||||||
|
|
||||||
segments.Clear();
|
segments.Clear();
|
||||||
segments.AddRange(Source.segments.Where(s => s.Length * drawSize.X > 1));
|
segments.AddRange(Source.segments.Where(s => s.Length * drawSize.X > 1));
|
||||||
|
|
||||||
|
tierColours.Clear();
|
||||||
|
tierColours.AddRange(Source.tierColours);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Draw(IRenderer renderer)
|
public override void Draw(IRenderer renderer)
|
||||||
@ -252,11 +241,27 @@ namespace osu.Game.Graphics.UserInterface
|
|||||||
Vector2Extensions.Transform(topRight, DrawInfo.Matrix),
|
Vector2Extensions.Transform(topRight, DrawInfo.Matrix),
|
||||||
Vector2Extensions.Transform(bottomLeft, DrawInfo.Matrix),
|
Vector2Extensions.Transform(bottomLeft, DrawInfo.Matrix),
|
||||||
Vector2Extensions.Transform(bottomRight, DrawInfo.Matrix)),
|
Vector2Extensions.Transform(bottomRight, DrawInfo.Matrix)),
|
||||||
Source.getSegmentColour(segment));
|
getSegmentColour(segment));
|
||||||
}
|
}
|
||||||
|
|
||||||
shader.Unbind();
|
shader.Unbind();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private ColourInfo getSegmentColour(SegmentInfo segment)
|
||||||
|
{
|
||||||
|
var segmentColour = new ColourInfo
|
||||||
|
{
|
||||||
|
TopLeft = DrawColourInfo.Colour.Interpolate(new Vector2(segment.Start, 0f)),
|
||||||
|
TopRight = DrawColourInfo.Colour.Interpolate(new Vector2(segment.End, 0f)),
|
||||||
|
BottomLeft = DrawColourInfo.Colour.Interpolate(new Vector2(segment.Start, 1f)),
|
||||||
|
BottomRight = DrawColourInfo.Colour.Interpolate(new Vector2(segment.End, 1f))
|
||||||
|
};
|
||||||
|
|
||||||
|
var tierColour = segment.Tier >= 0 ? tierColours[segment.Tier] : new Colour4(0, 0, 0, 0);
|
||||||
|
segmentColour.ApplyChild(tierColour);
|
||||||
|
|
||||||
|
return segmentColour;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected class SegmentManager : IEnumerable<SegmentInfo>
|
protected class SegmentManager : IEnumerable<SegmentInfo>
|
||||||
|
Loading…
Reference in New Issue
Block a user