1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-28 02:02:53 +08:00
osu-lazer/osu.Game/Rulesets/Objects
Bartłomiej Dach 7c9adc7ad3
Fix incorrect score conversion on selected beatmaps due to incorrect difficultyPeppyStars rounding
Fixes issue that occurs on *about* 246 beatmaps and was first described
by me on discord:

    https://discord.com/channels/188630481301012481/188630652340404224/1154367700378865715

and then rediscovered again during work on
https://github.com/ppy/osu/pull/26405:

    https://gist.github.com/bdach/414d5289f65b0399fa8f9732245a4f7c#venenog-on-ultmate-end-by-blacky-overdose-631

It so happens that in stable, due to .NET Framework internals, float
math would be performed using x87 registers and opcodes.
.NET (Core) however uses SSE instructions on 32- and 64-bit words.
x87 registers are _80 bits_ wide. Which is notably wider than _both_
float and double. Therefore, on a significant number of beatmaps,
the rounding would not produce correct values due to insufficient
precision.

See following gist for corroboration of the above:

    https://gist.github.com/bdach/dcde58d5a3607b0408faa3aa2b67bf10

Thus, to crudely - but, seemingly accurately, after checking across
all ranked maps - emulate this, use `decimal`, which is slow, but has
bigger precision than `double`. The single known exception beatmap
in whose case this results in an incorrect result is

    https://osu.ppy.sh/beatmapsets/1156087#osu/2625853

which is considered an "acceptable casualty" of sorts.

Doing this requires some fooling of the compiler / runtime (see second
inline comment in new method). To corroborate that this is required,
you can try the following code snippet:

    Console.WriteLine(string.Join(' ', BitConverter.GetBytes(1.3f).Select(x => x.ToString("X2"))));
    Console.WriteLine(string.Join(' ', BitConverter.GetBytes(1.3).Select(x => x.ToString("X2"))));
    Console.WriteLine();

    decimal d1 = (decimal)1.3f;
    decimal d2 = (decimal)1.3;
    decimal d3 = (decimal)(double)1.3f;

    Console.WriteLine(string.Join(' ', decimal.GetBits(d1).SelectMany(BitConverter.GetBytes).Select(x => x.ToString("X2"))));
    Console.WriteLine(string.Join(' ', decimal.GetBits(d2).SelectMany(BitConverter.GetBytes).Select(x => x.ToString("X2"))));
    Console.WriteLine(string.Join(' ', decimal.GetBits(d3).SelectMany(BitConverter.GetBytes).Select(x => x.ToString("X2"))));

which will print

    66 66 A6 3F
    CD CC CC CC CC CC F4 3F

    0D 00 00 00 00 00 00 00 00 00 00 00 00 00 01 00
    0D 00 00 00 00 00 00 00 00 00 00 00 00 00 01 00
    8C 5D 89 FB 3B 76 00 00 00 00 00 00 00 00 0E 00

Note that despite `d1` being converted from a less-precise floating-
-point value than `d2`, it still is represented 100% accurately as
a decimal number.

After applying this change, recomputation of legacy scoring attributes
for *all* rulesets will be required.
2024-01-10 19:30:18 +01:00
..
Drawables Reduce precision of audio balance adjustments during slider sliding 2024-01-05 02:26:30 +09:00
Legacy Fix incorrect score conversion on selected beatmaps due to incorrect difficultyPeppyStars rounding 2024-01-10 19:30:18 +01:00
Pooling Ensure synthetic entries from non-pooled DHO are linked to parents 2023-07-04 23:45:08 +02:00
Types Improve commenting around IHasCombo interfaces 2023-11-22 10:44:29 +09:00
BarLineGenerator.cs Move OmitFirstBarLine to TimingControlPoint 2023-02-28 19:29:31 +09:00
BezierConverter.cs Fix incorrect legacy conversion when B-splines are used 2023-11-20 15:08:58 +09:00
HitObject.cs Remove all usage of LegacyDifficultyControlPoint 2023-09-07 17:41:57 +09:00
HitObjectLifetimeEntry.cs Fix fallback for Judged to be more correct 2023-07-05 18:03:58 +09:00
HitObjectParser.cs Manual fixes to reduce warnings to zero 2023-06-24 01:52:53 +09:00
HitObjectProperty.cs Fix osu! and catch hitobjects no longer scaled to 1 by default 2022-07-19 07:00:13 +03:00
IBarLine.cs Remove redundant nullable suppression directives 2023-06-07 08:20:41 +03:00
PathControlPoint.cs Automated #nullable processing 2022-06-17 16:37:17 +09:00
SliderEventGenerator.cs Add better commenting around legacy last tick edge case 2023-10-04 13:45:26 +09:00
SliderPath.cs fix near-zero length sliders n stuff being placeable 2023-12-19 21:20:21 +01:00
SliderPathExtensions.cs Fix code style/quality issues 2023-11-13 08:25:27 +01:00
SyntheticHitObjectEntry.cs Automated pass 2023-06-24 01:00:03 +09:00