/* Copyright (c) 2015-2020 - present Advanced Micro Devices, Inc. All rights reserved. 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. */ #ifndef HIP_CODE_OBJECT_HPP #define HIP_CODE_OBJECT_HPP #include "hip_global.hpp" #include #include #include "hip/hip_runtime.h" #include "hip/hip_runtime_api.h" #include "hip_internal.hpp" #include "device/device.hpp" #include "platform/program.hpp" //Forward Declaration for friend usage class PlatformState; namespace hip { //Code Object base class class CodeObject { public: virtual ~CodeObject() {} // Functions to add_dev_prog and build static hipError_t add_program(int deviceId, hipModule_t hmod, const void* binary_ptr, size_t binary_size); static hipError_t build_module(hipModule_t hmod, const std::vector& devices); // Given an file desc and file size, extracts to code object for corresponding devices, // return code_objs{binary_ptr, binary_size}, which could be used to determine foffset static hipError_t ExtractCodeObjectFromFile(amd::Os::FileDesc fdesc, size_t fsize, const void ** image, const std::vector& device_names, std::vector>& code_objs); // Given an ptr to memory, extracts to code object for corresponding devices, // returns code_objs{binary_ptr, binary_size} and uniform resource indicator static hipError_t ExtractCodeObjectFromMemory(const void* data, const std::vector& device_names, std::vector>& code_objs, std::string& uri); static uint64_t ElfSize(const void* emi); protected: //Given an ptr to image or file, extracts to code object //for corresponding devices static hipError_t extractCodeObjectFromFatBinary(const void*, const std::vector&, std::vector>&); CodeObject() {} private: friend const std::vector& modules(); }; //Dynamic Code Object class DynCO : public CodeObject { amd::Monitor dclock_{"Guards Dynamic Code object", true}; public: DynCO() : device_id_(ihipGetDevice()) {} virtual ~DynCO(); //LoadsCodeObject and its data hipError_t loadCodeObject(const char* fname, const void* image=nullptr); hipModule_t module() { return fb_info_->Module(ihipGetDevice()); }; //Gets GlobalVar/Functions from a dynamically loaded code object hipError_t getDynFunc(hipFunction_t* hfunc, std::string func_name); hipError_t getDeviceVar(DeviceVar** dvar, std::string var_name); // Device ID Check to check if module is launched in the same device it was loaded. inline void CheckDeviceIdMatch() { if (device_id_ != ihipGetDevice()) { guarantee(false, "Device mismatch from where this module is loaded"); } } private: int device_id_; FatBinaryInfo* fb_info_; //Maps for vars/funcs, could be keyed in with std::string name std::unordered_map functions_; std::unordered_map vars_; //Populate Global Vars/Funcs from an code object(@ module_load) hipError_t populateDynGlobalFuncs(); hipError_t populateDynGlobalVars(); }; //Static Code Object class StatCO: public CodeObject { amd::Monitor sclock_{"Guards Static Code object", true}; public: StatCO(); virtual ~StatCO(); //Add/Remove/Digest Fat Binaries passed to us from "__hipRegisterFatBinary" FatBinaryInfo** addFatBinary(const void* data, bool initialized); hipError_t removeFatBinary(FatBinaryInfo** module); hipError_t digestFatBinary(const void* data, FatBinaryInfo*& programs); //Register vars/funcs given to use from __hipRegister[Var/Func/ManagedVar] hipError_t registerStatFunction(const void* hostFunction, Function* func); hipError_t registerStatGlobalVar(const void* hostVar, Var* var); hipError_t registerStatManagedVar(Var *var); //Retrive Vars/Funcs for a given hostSidePtr(const void*), unless stated otherwise. hipError_t getStatFunc(hipFunction_t* hfunc, const void* hostFunction, int deviceId); hipError_t getStatFuncAttr(hipFuncAttributes* func_attr, const void* hostFunction, int deviceId); hipError_t getStatGlobalVar(const void* hostVar, int deviceId, hipDeviceptr_t* dev_ptr, size_t* size_ptr); //Managed variable is a defined symbol in code object //pointer to the alocated managed memory has to be copied to the address of symbol hipError_t initStatManagedVarDevicePtr(int deviceId); private: friend class ::PlatformState; //Populated during __hipRegisterFatBinary std::unordered_map modules_; //Populated during __hipRegisterFuncs std::unordered_map functions_; //Populated during __hipRegisterVars std::unordered_map vars_; //Populated during __hipRegisterManagedVar std::vector managedVars_; std::unordered_map managedVarsDevicePtrInitalized_; }; }; // namespace hip #endif /* HIP_CODE_OBJECT_HPP */