| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- //
- // Copyright (c) 2008-2017 the Urho3D project.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- //
- #pragma once
- #include "../Container/HashSet.h"
- #include "../Container/List.h"
- #include "../Core/Mutex.h"
- #include "../IO/File.h"
- #include "../Resource/Resource.h"
- namespace Atomic
- {
- class BackgroundLoader;
- class FileWatcher;
- class PackageFile;
- /// Sets to priority so that a package or file is pushed to the end of the vector.
- static const unsigned PRIORITY_LAST = 0xffffffff;
- /// Container of resources with specific type.
- struct ResourceGroup
- {
- /// Construct with defaults.
- ResourceGroup() :
- memoryBudget_(0),
- memoryUse_(0)
- {
- }
- /// Memory budget.
- unsigned long long memoryBudget_;
- /// Current memory use.
- unsigned long long memoryUse_;
- /// Resources.
- HashMap<StringHash, SharedPtr<Resource> > resources_;
- };
- /// Resource request types.
- enum ResourceRequest
- {
- RESOURCE_CHECKEXISTS = 0,
- RESOURCE_GETFILE = 1
- };
- // ATOMIC BEGIN
- /// Optional resource request processor. Can deny requests, re-route resource file names, or perform other processing per request.
- class ATOMIC_API ResourceRouter : public Object
- {
- ATOMIC_OBJECT(ResourceRouter, Object);
- public:
- /// Construct.
- ResourceRouter(Context* context) :
- Object(context)
- {
- }
- /// Process the resource request and optionally modify the resource name string. Empty name string means the resource is not found or not allowed.
- virtual void Route(String& name, StringHash type, ResourceRequest requestType) = 0;
- };
- /// Helper class to expose resource iteration to script
- class ResourceNameIterator : public RefCounted
- {
- ATOMIC_REFCOUNTED(ResourceNameIterator);
- public:
- ResourceNameIterator();
- const String& GetCurrent();
- bool MoveNext();
- void Reset();
- Vector<String> filenames_;
- private:
- unsigned index_;
- };
- // ATOMIC END
- /// %Resource cache subsystem. Loads resources on demand and stores them for later access.
- class ATOMIC_API ResourceCache : public Object
- {
- ATOMIC_OBJECT(ResourceCache, Object);
- public:
- /// Construct.
- ResourceCache(Context* context);
- /// Destruct. Free all resources.
- virtual ~ResourceCache();
- /// Add a resource load directory. Optional priority parameter which will control search order.
- bool AddResourceDir(const String& pathName, unsigned priority = PRIORITY_LAST);
- /// Add a package file for loading resources from. Optional priority parameter which will control search order.
- bool AddPackageFile(PackageFile* package, unsigned priority = PRIORITY_LAST);
- /// Add a package file for loading resources from by name. Optional priority parameter which will control search order.
- bool AddPackageFile(const String& fileName, unsigned priority = PRIORITY_LAST);
- /// Add a manually created resource. Must be uniquely named within its type.
- bool AddManualResource(Resource* resource);
- /// Remove a resource load directory.
- void RemoveResourceDir(const String& pathName);
- /// Remove a package file. Optionally release the resources loaded from it.
- void RemovePackageFile(PackageFile* package, bool releaseResources = true, bool forceRelease = false);
- /// Remove a package file by name. Optionally release the resources loaded from it.
- void RemovePackageFile(const String& fileName, bool releaseResources = true, bool forceRelease = false);
- /// Release a resource by name.
- void ReleaseResource(StringHash type, const String& name, bool force = false);
- /// Release all resources of a specific type.
- void ReleaseResources(StringHash type, bool force = false);
- /// Release resources of a specific type and partial name.
- void ReleaseResources(StringHash type, const String& partialName, bool force = false);
- /// Release resources of all types by partial name.
- void ReleaseResources(const String& partialName, bool force = false);
- /// Release all resources. When called with the force flag false, releases all currently unused resources.
- void ReleaseAllResources(bool force = false);
- /// Reload a resource. Return true on success. The resource will not be removed from the cache in case of failure.
- bool ReloadResource(Resource* resource);
- /// Reload a resource based on filename. Causes also reload of dependent resources if necessary.
- void ReloadResourceWithDependencies(const String& fileName);
- /// Set memory budget for a specific resource type, default 0 is unlimited.
- void SetMemoryBudget(StringHash type, unsigned long long budget);
- /// Enable or disable automatic reloading of resources as files are modified. Default false.
- void SetAutoReloadResources(bool enable);
- /// Enable or disable returning resources that failed to load. Default false. This may be useful in editing to not lose resource ref attributes.
- void SetReturnFailedResources(bool enable) { returnFailedResources_ = enable; }
- /// Define whether when getting resources should check package files or directories first. True for packages, false for directories.
- void SetSearchPackagesFirst(bool value) { searchPackagesFirst_ = value; }
- /// Set how many milliseconds maximum per frame to spend on finishing background loaded resources.
- void SetFinishBackgroundResourcesMs(int ms) { finishBackgroundResourcesMs_ = Max(ms, 1); }
- /// Add a resource router object. By default there is none, so the routing process is skipped.
- void AddResourceRouter(ResourceRouter* router, bool addAsFirst = false);
- /// Remove a resource router object.
- void RemoveResourceRouter(ResourceRouter* router);
- // ATOMIC BEGIN
- /// Open and return a file from the resource load paths or from inside a package file. If not found, use a fallback search with absolute path. Return null if fails. Can be called from outside the main thread.
- SharedPtr<File> GetFile(const String& name, bool sendEventOnFailure = true, StringHash type = StringHash::ZERO);
- // ATOMIC END
- /// Return a resource by type and name. Load if not loaded yet. Return null if not found or if fails, unless SetReturnFailedResources(true) has been called. Can be called only from the main thread.
- Resource* GetResource(StringHash type, const String& name, bool sendEventOnFailure = true);
- /// Load a resource without storing it in the resource cache. Return null if not found or if fails. Can be called from outside the main thread if the resource itself is safe to load completely (it does not possess for example GPU data.)
- SharedPtr<Resource> GetTempResource(StringHash type, const String& name, bool sendEventOnFailure = true);
- /// Background load a resource. An event will be sent when complete. Return true if successfully stored to the load queue, false if eg. already exists. Can be called from outside the main thread.
- bool BackgroundLoadResource(StringHash type, const String& name, bool sendEventOnFailure = true, Resource* caller = 0);
- /// Return number of pending background-loaded resources.
- unsigned GetNumBackgroundLoadResources() const;
- /// Return all loaded resources of a specific type.
- void GetResources(PODVector<Resource*>& result, StringHash type) const;
- /// Return an already loaded resource of specific type & name, or null if not found. Will not load if does not exist.
- Resource* GetExistingResource(StringHash type, const String& name);
- /// Return all loaded resources.
- const HashMap<StringHash, ResourceGroup>& GetAllResources() const { return resourceGroups_; }
- /// Return added resource load directories.
- const Vector<String>& GetResourceDirs() const { return resourceDirs_; }
- /// Return added package files.
- const Vector<SharedPtr<PackageFile> >& GetPackageFiles() const { return packages_; }
- /// Template version of returning a resource by name.
- template <class T> T* GetResource(const String& name, bool sendEventOnFailure = true);
- /// Template version of returning an existing resource by name.
- template <class T> T* GetExistingResource(const String& name);
- /// Template version of loading a resource without storing it to the cache.
- template <class T> SharedPtr<T> GetTempResource(const String& name, bool sendEventOnFailure = true);
- /// Template version of releasing a resource by name.
- template <class T> void ReleaseResource(const String& name, bool force = false);
- /// Template version of queueing a resource background load.
- template <class T> bool BackgroundLoadResource(const String& name, bool sendEventOnFailure = true, Resource* caller = 0);
- /// Template version of returning loaded resources of a specific type.
- template <class T> void GetResources(PODVector<T*>& result) const;
- /// Return whether a file exists in the resource directories or package files. Does not check manually added in-memory resources.
- bool Exists(const String& name) const;
- /// Return memory budget for a resource type.
- unsigned long long GetMemoryBudget(StringHash type) const;
- /// Return total memory use for a resource type.
- unsigned long long GetMemoryUse(StringHash type) const;
- /// Return total memory use for all resources.
- unsigned long long GetTotalMemoryUse() const;
- /// Return full absolute file name of resource if possible, or empty if not found.
- String GetResourceFileName(const String& name) const;
- /// Return whether automatic resource reloading is enabled.
- bool GetAutoReloadResources() const { return autoReloadResources_; }
- /// Return whether resources that failed to load are returned.
- bool GetReturnFailedResources() const { return returnFailedResources_; }
- /// Return whether when getting resources should check package files or directories first.
- bool GetSearchPackagesFirst() const { return searchPackagesFirst_; }
- /// Return how many milliseconds maximum to spend on finishing background loaded resources.
- int GetFinishBackgroundResourcesMs() const { return finishBackgroundResourcesMs_; }
- /// Return a resource router by index.
- ResourceRouter* GetResourceRouter(unsigned index) const;
- /// Return either the path itself or its parent, based on which of them has recognized resource subdirectories.
- String GetPreferredResourceDir(const String& path) const;
- /// Remove unsupported constructs from the resource name to prevent ambiguity, and normalize absolute filename to resource path relative if possible.
- String SanitateResourceName(const String& name) const;
- /// Remove unnecessary constructs from a resource directory name and ensure it to be an absolute path.
- String SanitateResourceDirName(const String& name) const;
- /// Store a dependency for a resource. If a dependency file changes, the resource will be reloaded.
- void StoreResourceDependency(Resource* resource, const String& dependency);
- /// Reset dependencies for a resource.
- void ResetDependencies(Resource* resource);
- /// Returns a formatted string containing the memory actively used.
- String PrintMemoryUsage() const;
- // ATOMIC BEGIN
-
- /// Get the number of resource directories
- unsigned GetNumResourceDirs() const { return resourceDirs_.Size(); }
- /// Get resource directory at a given index
- const String& GetResourceDir(unsigned index) const { return index < resourceDirs_.Size() ? resourceDirs_[index] : String::EMPTY; }
-
- /// Scan for specified files.
- void Scan(Vector<String>& result, const String& pathName, const String& filter, unsigned flags, bool recursive) const;
- /// Scan specified files, returning them as an iterator
- SharedPtr<ResourceNameIterator> Scan(const String& pathName, const String& filter, unsigned flags, bool recursive) const;
- /// Returns a formatted string containing the currently loaded resources with optional type name filter.
- String PrintResources(const String& typeName = String::EMPTY) const;
- // ATOMIC END
- private:
- /// Find a resource.
- const SharedPtr<Resource>& FindResource(StringHash type, StringHash nameHash);
- /// Find a resource by name only. Searches all type groups.
- const SharedPtr<Resource>& FindResource(StringHash nameHash);
- /// Release resources loaded from a package file.
- void ReleasePackageResources(PackageFile* package, bool force = false);
- /// Update a resource group. Recalculate memory use and release resources if over memory budget.
- void UpdateResourceGroup(StringHash type);
- /// Handle begin frame event. Automatic resource reloads and the finalization of background loaded resources are processed here.
- void HandleBeginFrame(StringHash eventType, VariantMap& eventData);
- /// Search FileSystem for file.
- File* SearchResourceDirs(const String& nameIn);
- /// Search resource packages for file.
- File* SearchPackages(const String& nameIn);
- /// Mutex for thread-safe access to the resource directories, resource packages and resource dependencies.
- mutable Mutex resourceMutex_;
- /// Resources by type.
- HashMap<StringHash, ResourceGroup> resourceGroups_;
- /// Resource load directories.
- Vector<String> resourceDirs_;
- /// File watchers for resource directories, if automatic reloading enabled.
- Vector<SharedPtr<FileWatcher> > fileWatchers_;
- /// Package files.
- Vector<SharedPtr<PackageFile> > packages_;
- /// Dependent resources. Only used with automatic reload to eg. trigger reload of a cube texture when any of its faces change.
- HashMap<StringHash, HashSet<StringHash> > dependentResources_;
- /// Resource background loader.
- SharedPtr<BackgroundLoader> backgroundLoader_;
- /// Resource routers.
- Vector<SharedPtr<ResourceRouter> > resourceRouters_;
- /// Automatic resource reloading flag.
- bool autoReloadResources_;
- /// Return failed resources flag.
- bool returnFailedResources_;
- /// Search priority flag.
- bool searchPackagesFirst_;
- /// Resource routing flag to prevent endless recursion.
- mutable bool isRouting_;
- /// How many milliseconds maximum per frame to spend on finishing background loaded resources.
- int finishBackgroundResourcesMs_;
- };
- template <class T> T* ResourceCache::GetExistingResource(const String& name)
- {
- StringHash type = T::GetTypeStatic();
- return static_cast<T*>(GetExistingResource(type, name));
- }
- template <class T> T* ResourceCache::GetResource(const String& name, bool sendEventOnFailure)
- {
- StringHash type = T::GetTypeStatic();
- return static_cast<T*>(GetResource(type, name, sendEventOnFailure));
- }
- template <class T> void ResourceCache::ReleaseResource(const String& name, bool force)
- {
- StringHash type = T::GetTypeStatic();
- ReleaseResource(type, name, force);
- }
- template <class T> SharedPtr<T> ResourceCache::GetTempResource(const String& name, bool sendEventOnFailure)
- {
- StringHash type = T::GetTypeStatic();
- return StaticCast<T>(GetTempResource(type, name, sendEventOnFailure));
- }
- template <class T> bool ResourceCache::BackgroundLoadResource(const String& name, bool sendEventOnFailure, Resource* caller)
- {
- StringHash type = T::GetTypeStatic();
- return BackgroundLoadResource(type, name, sendEventOnFailure, caller);
- }
- template <class T> void ResourceCache::GetResources(PODVector<T*>& result) const
- {
- PODVector<Resource*>& resources = reinterpret_cast<PODVector<Resource*>&>(result);
- StringHash type = T::GetTypeStatic();
- GetResources(resources, type);
- // Perform conversion of the returned pointers
- for (unsigned i = 0; i < result.Size(); ++i)
- {
- Resource* resource = resources[i];
- result[i] = static_cast<T*>(resource);
- }
- }
- /// Register Resource library subsystems and objects.
- void ATOMIC_API RegisterResourceLibrary(Context* context);
- // ATOMIC BEGIN
- /// Extension used for package files
- extern ATOMIC_API const char* PAK_EXTENSION;
- // ATOMIC END
- }
|