// Copyright (c) 2022 PaddlePaddle Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #pragma once #ifdef PADDLE_WITH_CUSTOM_DEVICE #include "paddle/phi/backends/device_base.h" #include "paddle/phi/backends/device_ext.h" #include "paddle/phi/backends/event.h" #include "paddle/phi/backends/stream.h" #include "paddle/phi/common/place.h" #include "paddle/phi/backends/dynload/port.h" #include "paddle/phi/core/utils/rw_lock.h" namespace phi { class Device final { public: Device(size_t dev_id, DeviceInterface* impl) : dev_id_(dev_id), impl_(impl) {} // Stream // ! Create an asynchronous stream void CreateStream( stream::Stream* stream, const stream::Stream::Priority& priority = stream::Stream::Priority::kNormal, const stream::Stream::Flag& flag = stream::Stream::Flag::kDefaultFlag); // ! Destroys an asynchronous stream. void DestroyStream(stream::Stream* stream); // ! Waits for stream tasks to complete. void SynchronizeStream(const stream::Stream* stream); // ! Queries an asynchronous stream for completion status. bool QueryStream(const stream::Stream* stream); // ! Add a callback to a compute stream. void AddCallback(stream::Stream* stream, stream::Stream::Callback* callback); // Event // ! Create an event. void CreateEvent(event::Event* event, event::Event::Flag flags); // ! Destroy an event. void DestroyEvent(event::Event* event); // ! Records an event. void RecordEvent(const event::Event* event, const stream::Stream* stream); // ! Waits for event to complete. void SynchronizeEvent(const event::Event* event); // ! Queries an event for completion status. bool QueryEvent(const event::Event* event); // ! Make a compute stream wait on an event void StreamWaitEvent(const stream::Stream* stream, const event::Event* event); // Memory void MemoryCopyH2D(void* dst, const void* src, size_t size, const stream::Stream* stream = nullptr); void MemoryCopyD2H(void* dst, const void* src, size_t size, const stream::Stream* stream = nullptr); void MemoryCopyD2D(void* dst, const void* src, size_t size, const stream::Stream* stream = nullptr); void MemoryCopyP2P(const Place& dst_place, void* dst, const void* src, size_t size, const stream::Stream* stream = nullptr); void* MemoryAllocate(size_t size); void MemoryDeallocate(void* ptr, size_t size); void* MemoryAllocateHost(size_t size); void MemoryDeallocateHost(void* ptr, size_t size); void* MemoryAllocateUnified(size_t size); void MemoryDeallocateUnified(void* ptr, size_t size); void MemorySet(void* ptr, uint8_t value, size_t size); std::string Type(); private: size_t dev_id_; DeviceInterface* impl_; }; class DeviceManager { public: static bool Register(std::unique_ptr<DeviceInterface> device); static bool RegisterPinnedDevice(DeviceInterface* device); static Device* GetDeviceWithPlace(const Place& place); static std::vector<std::string> GetAllDeviceTypes(); static std::vector<std::string> GetAllCustomDeviceTypes(); static std::vector<std::string> GetAllDeviceList(); static std::vector<std::string> GetAllCustomDeviceList(); static bool HasDeviceType(const std::string& device_type); static bool IsCustom(const std::string& device_type); // platform & device static void Initialize(const std::string& device_type); static void Finalize(const std::string& device_type); static void SynchronizeDevice(const Place& place); static void InitDevice(const Place& place); static void DeInitDevice(const Place& place); static void SetDevice(const std::string& device_type, size_t device_id); static void SetDevice(const Place& place); static int GetDevice(const std::string& device_type); static size_t GetMinChunkSize(const Place& place); static size_t GetMaxChunkSize(const Place& place); static size_t GetMaxAllocSize(const Place& place); static size_t GetInitAllocSize(const Place& place); static size_t GetReallocSize(const Place& place); static size_t GetExtraPaddingSize(const Place& place); static void MemoryStats(const Place& place, size_t* total, size_t* free); static size_t GetDeviceCount(const std::string& device_type); static std::vector<size_t> GetDeviceList(const std::string& device_type); static void Clear(); private: DISABLE_COPY_AND_ASSIGN(DeviceManager); DeviceManager() {} static DeviceManager& Instance(); static DeviceInterface* GetDeviceInterfaceWithType( const std::string& device_type); std::unordered_map<std::string, std::unique_ptr<DeviceInterface>> device_impl_map_; std::unordered_map<std::string, std::vector<std::unique_ptr<Device>>> device_map_; }; std::vector<std::string> ListAllLibraries(const std::string& library_dir); void LoadCustomRuntimeLib(const std::string& dso_lib_path, void* dso_handle); void LoadCustomRuntimeLib(const CustomRuntimeParams& runtime_params, std::unique_ptr<C_DeviceInterface> device_interface, const std::string& dso_lib_path, void* dso_handle); class Registrar { public: template <typename DeviceT> explicit Registrar(DeviceT* device_ptr) { DeviceManager::Register(std::unique_ptr<DeviceT>(device_ptr)); } void Touch() {} }; } // namespace phi #endif