[kernel] Define param order between compilers

Fix issue in clang where args were inverted last to first due to the way
c++ implements function calls.
The function make_tuple, being a function, has undefined ordering of
resolution of the values of its parameters `Ps(init)` and would vary
between compilers. Namely clang would resolve them in order and msvc
would resolve them in reverse order.
This normally would not matter except for the fact that init maintains
a mutable state which is affected my the order of operations:
init.ordinal is a counter and also defines where in memory a value is
stored.
The C++ standard doesn't define an order of resolution of parameters in
a function but it will define an order in a brace-initializer.
Switching make_tuple for a brace-initializer enforces an order which is
the same between all compilers (tested gcc, clang and msvc).
Prior code was written to decrement ordinal due to the reverse
traversal.
This has been switched to incrementing thanks to the in-order
traversal.
This commit is contained in:
Sandy Carter 2019-07-11 22:00:07 -04:00 committed by Triang3l
parent 49e194009b
commit 432369ae84
1 changed files with 12 additions and 5 deletions

View File

@ -148,7 +148,7 @@ class Param {
protected: protected:
Param() : ordinal_(-1) {} Param() : ordinal_(-1) {}
explicit Param(Init& init) : ordinal_(--init.ordinal) {} explicit Param(Init& init) : ordinal_(init.ordinal++) {}
template <typename V> template <typename V>
void LoadValue(Init& init, V* out_value) { void LoadValue(Init& init, V* out_value) {
@ -519,10 +519,13 @@ xe::cpu::Export* RegisterExport(R (*fn)(Ps&...), const char* name,
++export_entry->function_data.call_count; ++export_entry->function_data.call_count;
Param::Init init = { Param::Init init = {
ppc_context, ppc_context,
sizeof...(Ps),
0, 0,
}; };
auto params = std::make_tuple<Ps...>(Ps(init)...); // Using braces initializer instead of make_tuple because braces
// enforce execution order across compilers.
// The make_tuple order is undefined per the C++ standard and
// cause inconsitencies between msvc and clang.
std::tuple<Ps...> params = {Ps(init)...};
if (export_entry->tags & xe::cpu::ExportTag::kLog && if (export_entry->tags & xe::cpu::ExportTag::kLog &&
(!(export_entry->tags & xe::cpu::ExportTag::kHighFrequency) || (!(export_entry->tags & xe::cpu::ExportTag::kHighFrequency) ||
cvars::log_high_frequency_kernel_calls)) { cvars::log_high_frequency_kernel_calls)) {
@ -554,9 +557,13 @@ xe::cpu::Export* RegisterExport(void (*fn)(Ps&...), const char* name,
++export_entry->function_data.call_count; ++export_entry->function_data.call_count;
Param::Init init = { Param::Init init = {
ppc_context, ppc_context,
sizeof...(Ps), 0,
}; };
auto params = std::make_tuple<Ps...>(Ps(init)...); // Using braces initializer instead of make_tuple because braces
// enforce execution order across compilers.
// The make_tuple order is undefined per the C++ standard and
// cause inconsitencies between msvc and clang.
std::tuple<Ps...> params = {Ps(init)...};
if (export_entry->tags & xe::cpu::ExportTag::kLog && if (export_entry->tags & xe::cpu::ExportTag::kLog &&
(!(export_entry->tags & xe::cpu::ExportTag::kHighFrequency) || (!(export_entry->tags & xe::cpu::ExportTag::kHighFrequency) ||
cvars::log_high_frequency_kernel_calls)) { cvars::log_high_frequency_kernel_calls)) {