/* Copyright (C) 2019 Wildfire Games. * This file is part of 0 A.D. * * 0 A.D. is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * 0 A.D. is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with 0 A.D. If not, see . */ #ifndef INCLUDED_LOADERTHUNKS #define INCLUDED_LOADERTHUNKS // VC7 warns if T::*func is not aligned to its size (4..16 bytes on IA-32). // this is a bug, since sizeof(void*) would be enough. MS says it won't // be fixed: see http://www.dotnet247.com/247reference/msgs/1/7782.aspx // we don't make sure alignment is acceptable because both 12 and 16 bytes // may be required and padding to LCM(12,16) bytes would be wasteful; // therefore, just disable the warning. #if MSC_VERSION #pragma warning(disable: 4121) #endif // does this return code indicate the coroutine yielded and // wasn't yet finished? static inline bool ldr_was_interrupted(int ret) { return (0 < ret && ret <= 100); } template struct MemFun_t { NONCOPYABLE(MemFun_t); public: T* const this_; int (T::*func)(void); MemFun_t(T* this__, int(T::*func_)(void)) : this_(this__), func(func_) {} }; template static int MemFunThunk(std::shared_ptr param, double UNUSED(time_left)) { MemFun_t* const mf = static_cast*>(param.get()); return (mf->this_->*mf->func)(); } template void RegMemFun(T* this_, int(T::*func)(void), const wchar_t* description, int estimated_duration_ms) { LDR_Register(MemFunThunk, std::make_shared>(this_, func), description, estimated_duration_ms); } //////////////////////////////////////////////////////// template struct MemFun1_t { NONCOPYABLE(MemFun1_t); public: T* const this_; Arg arg; int (T::*func)(Arg); MemFun1_t(T* this__, int(T::*func_)(Arg), Arg arg_) : this_(this__), func(func_), arg(arg_) {} }; template static int MemFun1Thunk(shared_ptr param, double UNUSED(time_left)) { MemFun1_t* const mf = static_cast*>(param.get()); return (mf->this_->*mf->func)(mf->arg); } template void RegMemFun1(T* this_, int(T::*func)(Arg), Arg arg, const wchar_t* description, int estimated_duration_ms) { LDR_Register(MemFun1Thunk, std::make_shared >(this_, func, arg), description, estimated_duration_ms); } #endif // INCLUDED_LOADERTHUNKS