2021-09-30 06:40:41 +00:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
2019-01-24 08:43:03 +00:00
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
using System ;
2021-04-17 15:47:13 +00:00
using System.IO ;
2021-10-04 08:00:22 +00:00
using System.Linq ;
2021-09-30 06:40:41 +00:00
using JetBrains.Annotations ;
using osu.Framework.Audio ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Audio.Track ;
using osu.Framework.Graphics.Textures ;
2021-09-30 06:40:41 +00:00
using osu.Framework.IO.Stores ;
using osu.Framework.Lists ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Logging ;
2021-09-30 06:40:41 +00:00
using osu.Framework.Platform ;
using osu.Framework.Statistics ;
2020-10-16 05:39:02 +00:00
using osu.Framework.Testing ;
2018-04-13 09:19:50 +00:00
using osu.Game.Beatmaps.Formats ;
2019-09-09 22:43:30 +00:00
using osu.Game.IO ;
2018-04-13 09:19:50 +00:00
using osu.Game.Skinning ;
using osu.Game.Storyboards ;
namespace osu.Game.Beatmaps
{
2021-09-30 07:45:32 +00:00
public class WorkingBeatmapCache : IBeatmapResourceProvider , IWorkingBeatmapCache
2018-04-13 09:19:50 +00:00
{
2021-09-30 06:40:41 +00:00
private readonly WeakList < BeatmapManagerWorkingBeatmap > workingCache = new WeakList < BeatmapManagerWorkingBeatmap > ( ) ;
/// <summary>
/// A default representation of a WorkingBeatmap to use when no beatmap is available.
/// </summary>
public readonly WorkingBeatmap DefaultBeatmap ;
2021-09-30 07:45:32 +00:00
public BeatmapModelManager BeatmapManager { private get ; set ; }
2021-09-30 06:40:41 +00:00
private readonly AudioManager audioManager ;
private readonly IResourceStore < byte [ ] > resources ;
private readonly LargeTextureStore largeTextureStore ;
private readonly ITrackStore trackStore ;
private readonly IResourceStore < byte [ ] > files ;
[CanBeNull]
private readonly GameHost host ;
2021-11-09 08:27:07 +00:00
public WorkingBeatmapCache ( ITrackStore trackStore , AudioManager audioManager , IResourceStore < byte [ ] > resources , IResourceStore < byte [ ] > files , WorkingBeatmap defaultBeatmap = null , GameHost host = null )
2021-09-30 06:40:41 +00:00
{
DefaultBeatmap = defaultBeatmap ;
this . audioManager = audioManager ;
this . resources = resources ;
this . host = host ;
this . files = files ;
largeTextureStore = new LargeTextureStore ( host ? . CreateTextureLoaderStore ( files ) ) ;
2021-11-09 08:27:07 +00:00
this . trackStore = trackStore ;
2021-09-30 06:40:41 +00:00
}
public void Invalidate ( BeatmapSetInfo info )
{
if ( info . Beatmaps = = null ) return ;
foreach ( var b in info . Beatmaps )
Invalidate ( b ) ;
}
public void Invalidate ( BeatmapInfo info )
{
lock ( workingCache )
{
2021-11-24 03:49:57 +00:00
var working = workingCache . FirstOrDefault ( w = > info . Equals ( w . BeatmapInfo ) ) ;
2021-10-14 04:58:36 +00:00
2021-09-30 06:40:41 +00:00
if ( working ! = null )
2021-10-14 04:58:36 +00:00
{
Logger . Log ( $"Invalidating working beatmap cache for {info}" ) ;
2021-09-30 06:40:41 +00:00
workingCache . Remove ( working ) ;
2021-10-14 04:58:36 +00:00
}
2021-09-30 06:40:41 +00:00
}
}
public virtual WorkingBeatmap GetWorkingBeatmap ( BeatmapInfo beatmapInfo )
{
// if there are no files, presume the full beatmap info has not yet been fetched from the database.
if ( beatmapInfo ? . BeatmapSet ? . Files . Count = = 0 )
{
int lookupId = beatmapInfo . ID ;
beatmapInfo = BeatmapManager . QueryBeatmap ( b = > b . ID = = lookupId ) ;
}
if ( beatmapInfo ? . BeatmapSet = = null )
return DefaultBeatmap ;
lock ( workingCache )
{
2021-11-24 03:49:57 +00:00
var working = workingCache . FirstOrDefault ( w = > beatmapInfo . Equals ( w . BeatmapInfo ) ) ;
2021-10-14 04:58:36 +00:00
2021-09-30 06:40:41 +00:00
if ( working ! = null )
return working ;
beatmapInfo . Metadata ? ? = beatmapInfo . BeatmapSet . Metadata ;
workingCache . Add ( working = new BeatmapManagerWorkingBeatmap ( beatmapInfo , this ) ) ;
// best effort; may be higher than expected.
GlobalStatistics . Get < int > ( nameof ( Beatmaps ) , $"Cached {nameof(WorkingBeatmap)}s" ) . Value = workingCache . Count ( ) ;
return working ;
}
}
#region IResourceStorageProvider
TextureStore IBeatmapResourceProvider . LargeTextureStore = > largeTextureStore ;
ITrackStore IBeatmapResourceProvider . Tracks = > trackStore ;
AudioManager IStorageResourceProvider . AudioManager = > audioManager ;
IResourceStore < byte [ ] > IStorageResourceProvider . Files = > files ;
IResourceStore < byte [ ] > IStorageResourceProvider . Resources = > resources ;
IResourceStore < TextureUpload > IStorageResourceProvider . CreateTextureLoaderStore ( IResourceStore < byte [ ] > underlyingStore ) = > host ? . CreateTextureLoaderStore ( underlyingStore ) ;
#endregion
2020-10-16 05:39:02 +00:00
[ExcludeFromDynamicCompile]
2020-08-11 04:48:57 +00:00
private class BeatmapManagerWorkingBeatmap : WorkingBeatmap
2018-04-13 09:19:50 +00:00
{
2020-12-22 03:06:10 +00:00
[NotNull]
2020-12-21 05:06:50 +00:00
private readonly IBeatmapResourceProvider resources ;
2018-04-13 09:19:50 +00:00
2020-12-22 03:06:10 +00:00
public BeatmapManagerWorkingBeatmap ( BeatmapInfo beatmapInfo , [ NotNull ] IBeatmapResourceProvider resources )
: base ( beatmapInfo , resources . AudioManager )
2018-04-13 09:19:50 +00:00
{
2020-12-21 05:06:50 +00:00
this . resources = resources ;
2018-04-13 09:19:50 +00:00
}
2018-04-19 11:44:38 +00:00
protected override IBeatmap GetBeatmap ( )
2018-04-13 09:19:50 +00:00
{
2020-08-24 10:38:05 +00:00
if ( BeatmapInfo . Path = = null )
2020-09-04 04:13:53 +00:00
return new Beatmap { BeatmapInfo = BeatmapInfo } ;
2020-08-24 10:38:05 +00:00
2018-04-13 09:19:50 +00:00
try
{
2021-04-17 15:49:10 +00:00
using ( var stream = new LineBufferedReader ( GetStream ( BeatmapSetInfo . GetPathForFile ( BeatmapInfo . Path ) ) ) )
2018-04-13 09:19:50 +00:00
return Decoder . GetDecoder < Beatmap > ( stream ) . Decode ( stream ) ;
}
2020-02-10 08:25:11 +00:00
catch ( Exception e )
2018-04-13 09:19:50 +00:00
{
2020-02-10 08:25:11 +00:00
Logger . Error ( e , "Beatmap failed to load" ) ;
2018-04-13 09:19:50 +00:00
return null ;
}
}
2018-09-06 04:15:43 +00:00
protected override bool BackgroundStillValid ( Texture b ) = > false ; // bypass lazy logic. we want to return a new background each time for refcounting purposes.
2018-04-13 09:19:50 +00:00
protected override Texture GetBackground ( )
{
2021-11-04 05:11:21 +00:00
if ( string . IsNullOrEmpty ( Metadata ? . BackgroundFile ) )
2018-04-13 09:19:50 +00:00
return null ;
try
{
2021-04-17 15:49:10 +00:00
return resources . LargeTextureStore . Get ( BeatmapSetInfo . GetPathForFile ( Metadata . BackgroundFile ) ) ;
2018-04-13 09:19:50 +00:00
}
2020-02-10 08:25:11 +00:00
catch ( Exception e )
2018-04-13 09:19:50 +00:00
{
2020-02-10 08:25:11 +00:00
Logger . Error ( e , "Background failed to load" ) ;
2018-04-13 09:19:50 +00:00
return null ;
}
}
2019-08-30 20:19:34 +00:00
2020-08-07 13:31:41 +00:00
protected override Track GetBeatmapTrack ( )
2018-04-13 09:19:50 +00:00
{
2021-11-04 05:01:01 +00:00
if ( string . IsNullOrEmpty ( Metadata ? . AudioFile ) )
2020-09-01 06:48:13 +00:00
return null ;
2018-04-13 09:19:50 +00:00
try
{
2021-04-17 15:49:10 +00:00
return resources . Tracks . Get ( BeatmapSetInfo . GetPathForFile ( Metadata . AudioFile ) ) ;
2018-04-13 09:19:50 +00:00
}
2020-02-10 08:25:11 +00:00
catch ( Exception e )
2018-04-13 09:19:50 +00:00
{
2020-02-10 08:25:11 +00:00
Logger . Error ( e , "Track failed to load" ) ;
2018-06-27 07:02:49 +00:00
return null ;
2018-04-13 09:19:50 +00:00
}
}
2018-06-27 07:07:18 +00:00
protected override Waveform GetWaveform ( )
{
2021-11-04 05:01:01 +00:00
if ( string . IsNullOrEmpty ( Metadata ? . AudioFile ) )
2020-09-01 06:48:13 +00:00
return null ;
2018-06-27 07:07:18 +00:00
try
{
2021-04-17 15:49:10 +00:00
var trackData = GetStream ( BeatmapSetInfo . GetPathForFile ( Metadata . AudioFile ) ) ;
2018-06-27 07:07:18 +00:00
return trackData = = null ? null : new Waveform ( trackData ) ;
}
2020-02-10 08:25:11 +00:00
catch ( Exception e )
2018-06-27 07:07:18 +00:00
{
2020-02-10 08:25:11 +00:00
Logger . Error ( e , "Waveform failed to load" ) ;
2018-06-27 07:07:18 +00:00
return null ;
}
}
2018-04-13 09:19:50 +00:00
protected override Storyboard GetStoryboard ( )
{
Storyboard storyboard ;
2019-04-01 03:16:05 +00:00
2018-04-13 09:19:50 +00:00
try
{
2021-04-17 15:49:10 +00:00
using ( var stream = new LineBufferedReader ( GetStream ( BeatmapSetInfo . GetPathForFile ( BeatmapInfo . Path ) ) ) )
2018-04-13 09:19:50 +00:00
{
var decoder = Decoder . GetDecoder < Storyboard > ( stream ) ;
2021-10-27 04:04:41 +00:00
string storyboardFilename = BeatmapSetInfo ? . Files . FirstOrDefault ( f = > f . Filename . EndsWith ( ".osb" , StringComparison . OrdinalIgnoreCase ) ) ? . Filename ;
2021-10-04 07:50:29 +00:00
2018-04-13 09:19:50 +00:00
// todo: support loading from both set-wide storyboard *and* beatmap specific.
2021-10-04 07:50:29 +00:00
if ( string . IsNullOrEmpty ( storyboardFilename ) )
2018-04-13 09:19:50 +00:00
storyboard = decoder . Decode ( stream ) ;
else
{
2021-10-04 07:50:29 +00:00
using ( var secondaryStream = new LineBufferedReader ( GetStream ( BeatmapSetInfo . GetPathForFile ( storyboardFilename ) ) ) )
2018-04-13 09:19:50 +00:00
storyboard = decoder . Decode ( stream , secondaryStream ) ;
}
}
}
catch ( Exception e )
{
Logger . Error ( e , "Storyboard failed to load" ) ;
storyboard = new Storyboard ( ) ;
}
storyboard . BeatmapInfo = BeatmapInfo ;
return storyboard ;
}
2021-08-15 16:38:01 +00:00
protected internal override ISkin GetSkin ( )
2018-04-13 09:19:50 +00:00
{
try
{
2020-12-21 06:14:32 +00:00
return new LegacyBeatmapSkin ( BeatmapInfo , resources . Files , resources ) ;
2018-04-13 09:19:50 +00:00
}
catch ( Exception e )
{
Logger . Error ( e , "Skin failed to load" ) ;
2019-08-26 05:25:35 +00:00
return null ;
2018-04-13 09:19:50 +00:00
}
}
2021-04-17 15:47:13 +00:00
public override Stream GetStream ( string storagePath ) = > resources . Files . GetStream ( storagePath ) ;
2018-04-13 09:19:50 +00:00
}
}
}