2017-02-10 15:05:56 +08:00
|
|
|
//===- X86LegalizerInfo.cpp --------------------------------------*- C++ -*-==//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// \file
|
|
|
|
/// This file implements the targeting of the Machinelegalizer class for X86.
|
|
|
|
/// \todo This should be generated by TableGen.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "X86LegalizerInfo.h"
|
|
|
|
#include "X86Subtarget.h"
|
2017-03-26 16:11:12 +08:00
|
|
|
#include "X86TargetMachine.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetOpcodes.h"
|
2018-03-30 01:21:10 +08:00
|
|
|
#include "llvm/CodeGen/ValueTypes.h"
|
2017-02-10 15:05:56 +08:00
|
|
|
#include "llvm/IR/DerivedTypes.h"
|
|
|
|
#include "llvm/IR/Type.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
2017-03-03 16:06:46 +08:00
|
|
|
using namespace TargetOpcode;
|
2018-01-30 01:37:29 +08:00
|
|
|
using namespace LegalizeActions;
|
2017-02-10 15:05:56 +08:00
|
|
|
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
/// FIXME: The following static functions are SizeChangeStrategy functions
|
|
|
|
/// that are meant to temporarily mimic the behaviour of the old legalization
|
|
|
|
/// based on doubling/halving non-legal types as closely as possible. This is
|
|
|
|
/// not entirly possible as only legalizing the types that are exactly a power
|
|
|
|
/// of 2 times the size of the legal types would require specifying all those
|
|
|
|
/// sizes explicitly.
|
|
|
|
/// In practice, not specifying those isn't a problem, and the below functions
|
|
|
|
/// should disappear quickly as we add support for legalizing non-power-of-2
|
|
|
|
/// sized types further.
|
|
|
|
static void
|
|
|
|
addAndInterleaveWithUnsupported(LegalizerInfo::SizeAndActionsVec &result,
|
|
|
|
const LegalizerInfo::SizeAndActionsVec &v) {
|
|
|
|
for (unsigned i = 0; i < v.size(); ++i) {
|
|
|
|
result.push_back(v[i]);
|
|
|
|
if (i + 1 < v[i].first && i + 1 < v.size() &&
|
|
|
|
v[i + 1].first != v[i].first + 1)
|
2018-01-30 01:37:29 +08:00
|
|
|
result.push_back({v[i].first + 1, Unsupported});
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static LegalizerInfo::SizeAndActionsVec
|
|
|
|
widen_1(const LegalizerInfo::SizeAndActionsVec &v) {
|
|
|
|
assert(v.size() >= 1);
|
|
|
|
assert(v[0].first > 1);
|
2018-01-30 01:37:29 +08:00
|
|
|
LegalizerInfo::SizeAndActionsVec result = {{1, WidenScalar},
|
|
|
|
{2, Unsupported}};
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
addAndInterleaveWithUnsupported(result, v);
|
|
|
|
auto Largest = result.back().first;
|
2018-01-30 01:37:29 +08:00
|
|
|
result.push_back({Largest + 1, Unsupported});
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-03-26 16:11:12 +08:00
|
|
|
X86LegalizerInfo::X86LegalizerInfo(const X86Subtarget &STI,
|
|
|
|
const X86TargetMachine &TM)
|
|
|
|
: Subtarget(STI), TM(TM) {
|
2017-02-10 15:05:56 +08:00
|
|
|
|
|
|
|
setLegalizerInfo32bit();
|
|
|
|
setLegalizerInfo64bit();
|
2017-03-03 16:06:46 +08:00
|
|
|
setLegalizerInfoSSE1();
|
|
|
|
setLegalizerInfoSSE2();
|
2017-05-08 17:03:37 +08:00
|
|
|
setLegalizerInfoSSE41();
|
2017-05-23 16:23:51 +08:00
|
|
|
setLegalizerInfoAVX();
|
2017-05-08 17:03:37 +08:00
|
|
|
setLegalizerInfoAVX2();
|
|
|
|
setLegalizerInfoAVX512();
|
|
|
|
setLegalizerInfoAVX512DQ();
|
|
|
|
setLegalizerInfoAVX512BW();
|
2017-02-10 15:05:56 +08:00
|
|
|
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
setLegalizeScalarToDifferentSizeStrategy(G_PHI, 0, widen_1);
|
|
|
|
for (unsigned BinOp : {G_SUB, G_MUL, G_AND, G_OR, G_XOR})
|
|
|
|
setLegalizeScalarToDifferentSizeStrategy(BinOp, 0, widen_1);
|
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE})
|
|
|
|
setLegalizeScalarToDifferentSizeStrategy(MemOp, 0,
|
|
|
|
narrowToSmallerAndWidenToSmallest);
|
|
|
|
setLegalizeScalarToDifferentSizeStrategy(
|
|
|
|
G_GEP, 1, widenToLargerTypesUnsupportedOtherwise);
|
|
|
|
setLegalizeScalarToDifferentSizeStrategy(
|
|
|
|
G_CONSTANT, 0, widenToLargerTypesAndNarrowToLargest);
|
|
|
|
|
2017-02-10 15:05:56 +08:00
|
|
|
computeTables();
|
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfo32bit() {
|
|
|
|
|
2018-03-14 08:36:23 +08:00
|
|
|
const LLT p0 = LLT::pointer(0, TM.getPointerSizeInBits(0));
|
2017-04-07 22:41:59 +08:00
|
|
|
const LLT s1 = LLT::scalar(1);
|
2017-02-10 15:05:56 +08:00
|
|
|
const LLT s8 = LLT::scalar(8);
|
|
|
|
const LLT s16 = LLT::scalar(16);
|
|
|
|
const LLT s32 = LLT::scalar(32);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT s64 = LLT::scalar(64);
|
2018-02-09 06:41:47 +08:00
|
|
|
const LLT s128 = LLT::scalar(128);
|
2017-02-10 15:05:56 +08:00
|
|
|
|
2017-08-24 15:06:27 +08:00
|
|
|
for (auto Ty : {p0, s1, s8, s16, s32})
|
|
|
|
setAction({G_IMPLICIT_DEF, Ty}, Legal);
|
|
|
|
|
2017-09-04 17:06:45 +08:00
|
|
|
for (auto Ty : {s8, s16, s32, p0})
|
|
|
|
setAction({G_PHI, Ty}, Legal);
|
|
|
|
|
[GlobalISel] Enable legalizing non-power-of-2 sized types.
This changes the interface of how targets describe how to legalize, see
the below description.
1. Interface for targets to describe how to legalize.
In GlobalISel, the API in the LegalizerInfo class is the main interface
for targets to specify which types are legal for which operations, and
what to do to turn illegal type/operation combinations into legal ones.
For each operation the type sizes that can be legalized without having
to change the size of the type are specified with a call to setAction.
This isn't different to how GlobalISel worked before. For example, for a
target that supports 32 and 64 bit adds natively:
for (auto Ty : {s32, s64})
setAction({G_ADD, 0, s32}, Legal);
or for a target that needs a library call for a 32 bit division:
setAction({G_SDIV, s32}, Libcall);
The main conceptual change to the LegalizerInfo API, is in specifying
how to legalize the type sizes for which a change of size is needed. For
example, in the above example, how to specify how all types from i1 to
i8388607 (apart from s32 and s64 which are legal) need to be legalized
and expressed in terms of operations on the available legal sizes
(again, i32 and i64 in this case). Before, the implementation only
allowed specifying power-of-2-sized types (e.g. setAction({G_ADD, 0,
s128}, NarrowScalar). A worse limitation was that if you'd wanted to
specify how to legalize all the sized types as allowed by the LLVM-IR
LangRef, i1 to i8388607, you'd have to call setAction 8388607-3 times
and probably would need a lot of memory to store all of these
specifications.
Instead, the legalization actions that need to change the size of the
type are specified now using a "SizeChangeStrategy". For example:
setLegalizeScalarToDifferentSizeStrategy(
G_ADD, 0, widenToLargerAndNarrowToLargest);
This example indicates that for type sizes for which there is a larger
size that can be legalized towards, do it by Widening the size.
For example, G_ADD on s17 will be legalized by first doing WidenScalar
to make it s32, after which it's legal.
The "NarrowToLargest" indicates what to do if there is no larger size
that can be legalized towards. E.g. G_ADD on s92 will be legalized by
doing NarrowScalar to s64.
Another example, taken from the ARM backend is:
for (unsigned Op : {G_SDIV, G_UDIV}) {
setLegalizeScalarToDifferentSizeStrategy(Op, 0,
widenToLargerTypesUnsupportedOtherwise);
if (ST.hasDivideInARMMode())
setAction({Op, s32}, Legal);
else
setAction({Op, s32}, Libcall);
}
For this example, G_SDIV on s8, on a target without a divide
instruction, would be legalized by first doing action (WidenScalar,
s32), followed by (Libcall, s32).
The same principle is also followed for when the number of vector lanes
on vector data types need to be changed, e.g.:
setAction({G_ADD, LLT::vector(8, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(16, 8)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(8, 16)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(2, 32)}, LegalizerInfo::Legal);
setAction({G_ADD, LLT::vector(4, 32)}, LegalizerInfo::Legal);
setLegalizeVectorElementToDifferentSizeStrategy(
G_ADD, 0, widenToLargerTypesUnsupportedOtherwise);
As currently implemented here, vector types are legalized by first
making the vector element size legal, followed by then making the number
of lanes legal. The strategy to follow in the first step is set by a
call to setLegalizeVectorElementToDifferentSizeStrategy, see example
above. The strategy followed in the second step
"moreToWiderTypesAndLessToWidest" (see code for its definition),
indicating that vectors are widened to more elements so they map to
natively supported vector widths, or when there isn't a legal wider
vector, split the vector to map it to the widest vector supported.
Therefore, for the above specification, some example legalizations are:
* getAction({G_ADD, LLT::vector(3, 3)})
returns {WidenScalar, LLT::vector(3, 8)}
* getAction({G_ADD, LLT::vector(3, 8)})
then returns {MoreElements, LLT::vector(8, 8)}
* getAction({G_ADD, LLT::vector(20, 8)})
returns {FewerElements, LLT::vector(16, 8)}
2. Key implementation aspects.
How to legalize a specific (operation, type index, size) tuple is
represented by mapping intervals of integers representing a range of
size types to an action to take, e.g.:
setScalarAction({G_ADD, LLT:scalar(1)},
{{1, WidenScalar}, // bit sizes [ 1, 31[
{32, Legal}, // bit sizes [32, 33[
{33, WidenScalar}, // bit sizes [33, 64[
{64, Legal}, // bit sizes [64, 65[
{65, NarrowScalar} // bit sizes [65, +inf[
});
Please note that most of the code to do the actual lowering of
non-power-of-2 sized types is currently missing, this is just trying to
make it possible for targets to specify what is legal, and how non-legal
types should be legalized. Probably quite a bit of further work is
needed in the actual legalizing and the other passes in GlobalISel to
support non-power-of-2 sized types.
I hope the documentation in LegalizerInfo.h and the examples provided in the
various {Target}LegalizerInfo.cpp and LegalizerInfoTest.cpp explains well
enough how this is meant to be used.
This drops the need for LLT::{half,double}...Size().
Differential Revision: https://reviews.llvm.org/D30529
llvm-svn: 317560
2017-11-07 18:34:34 +08:00
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB, G_MUL, G_AND, G_OR, G_XOR})
|
2017-03-23 23:25:57 +08:00
|
|
|
for (auto Ty : {s8, s16, s32})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
|
|
|
|
2017-05-17 20:48:08 +08:00
|
|
|
for (unsigned Op : {G_UADDE}) {
|
|
|
|
setAction({Op, s32}, Legal);
|
|
|
|
setAction({Op, 1, s1}, Legal);
|
|
|
|
}
|
|
|
|
|
2017-03-23 23:25:57 +08:00
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE}) {
|
|
|
|
for (auto Ty : {s8, s16, s32, p0})
|
|
|
|
setAction({MemOp, Ty}, Legal);
|
|
|
|
|
|
|
|
// And everything's fine in addrspace 0.
|
|
|
|
setAction({MemOp, 1, p0}, Legal);
|
2017-02-22 20:25:09 +08:00
|
|
|
}
|
2017-03-26 16:11:12 +08:00
|
|
|
|
|
|
|
// Pointer-handling
|
|
|
|
setAction({G_FRAME_INDEX, p0}, Legal);
|
2017-07-02 16:58:29 +08:00
|
|
|
setAction({G_GLOBAL_VALUE, p0}, Legal);
|
2017-04-07 22:41:59 +08:00
|
|
|
|
2017-05-08 17:40:43 +08:00
|
|
|
setAction({G_GEP, p0}, Legal);
|
|
|
|
setAction({G_GEP, 1, s32}, Legal);
|
|
|
|
|
2018-02-28 20:11:53 +08:00
|
|
|
if (!Subtarget.is64Bit()) {
|
2018-02-28 17:18:47 +08:00
|
|
|
getActionDefinitionsBuilder(G_PTRTOINT)
|
|
|
|
.legalForCartesianProduct({s1, s8, s16, s32}, {p0})
|
|
|
|
.maxScalar(0, s32)
|
|
|
|
.widenScalarToNextPow2(0, /*Min*/ 8);
|
2018-02-28 20:11:53 +08:00
|
|
|
getActionDefinitionsBuilder(G_INTTOPTR).legalFor({s32, p0});
|
[GlobalISel][X86] Support G_LSHR/G_ASHR/G_SHL
Support G_LSHR/G_ASHR/G_SHL. We have 3 variance for
shift instructions : shift gpr, shift imm, shift 1.
Currently GlobalIsel TableGen generate patterns for
shift imm and shift 1, but with shiftCount i8.
In G_LSHR/G_ASHR/G_SHL like LLVM-IR both arguments
has the same type, so for now only shift i8 can use
auto generated TableGen patterns.
The support of G_SHL/G_ASHR enables tryCombineSExt
from LegalizationArtifactCombiner.h to hit, which
results in different legalization for the following tests:
LLVM :: CodeGen/X86/GlobalISel/ext-x86-64.ll
LLVM :: CodeGen/X86/GlobalISel/gep.ll
LLVM :: CodeGen/X86/GlobalISel/legalize-ext-x86-64.mir
-; X64-NEXT: movsbl %dil, %eax
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: shll %cl, %edi
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: sarl %cl, %edi
+; X64-NEXT: movl %edi, %eax
..which is not optimal and should be addressed later.
Rework of the patch by igorb
Reviewed By: igorb
Differential Revision: https://reviews.llvm.org/D44395
llvm-svn: 327499
2018-03-14 19:23:57 +08:00
|
|
|
|
2018-03-14 23:41:11 +08:00
|
|
|
// Shifts and SDIV
|
|
|
|
getActionDefinitionsBuilder({G_SHL, G_LSHR, G_ASHR, G_SDIV})
|
[GlobalISel][X86] Support G_LSHR/G_ASHR/G_SHL
Support G_LSHR/G_ASHR/G_SHL. We have 3 variance for
shift instructions : shift gpr, shift imm, shift 1.
Currently GlobalIsel TableGen generate patterns for
shift imm and shift 1, but with shiftCount i8.
In G_LSHR/G_ASHR/G_SHL like LLVM-IR both arguments
has the same type, so for now only shift i8 can use
auto generated TableGen patterns.
The support of G_SHL/G_ASHR enables tryCombineSExt
from LegalizationArtifactCombiner.h to hit, which
results in different legalization for the following tests:
LLVM :: CodeGen/X86/GlobalISel/ext-x86-64.ll
LLVM :: CodeGen/X86/GlobalISel/gep.ll
LLVM :: CodeGen/X86/GlobalISel/legalize-ext-x86-64.mir
-; X64-NEXT: movsbl %dil, %eax
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: shll %cl, %edi
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: sarl %cl, %edi
+; X64-NEXT: movl %edi, %eax
..which is not optimal and should be addressed later.
Rework of the patch by igorb
Reviewed By: igorb
Differential Revision: https://reviews.llvm.org/D44395
llvm-svn: 327499
2018-03-14 19:23:57 +08:00
|
|
|
.legalFor({s8, s16, s32})
|
|
|
|
.clampScalar(0, s8, s32);
|
2018-02-28 20:11:53 +08:00
|
|
|
}
|
2018-02-28 17:18:47 +08:00
|
|
|
|
2017-08-21 18:51:54 +08:00
|
|
|
// Control-flow
|
|
|
|
setAction({G_BRCOND, s1}, Legal);
|
|
|
|
|
2017-04-07 22:41:59 +08:00
|
|
|
// Constants
|
|
|
|
for (auto Ty : {s8, s16, s32, p0})
|
|
|
|
setAction({TargetOpcode::G_CONSTANT, Ty}, Legal);
|
|
|
|
|
2017-05-01 14:30:16 +08:00
|
|
|
// Extensions
|
2017-07-10 17:07:34 +08:00
|
|
|
for (auto Ty : {s8, s16, s32}) {
|
|
|
|
setAction({G_ZEXT, Ty}, Legal);
|
|
|
|
setAction({G_SEXT, Ty}, Legal);
|
2017-09-11 17:41:13 +08:00
|
|
|
setAction({G_ANYEXT, Ty}, Legal);
|
2017-07-10 17:07:34 +08:00
|
|
|
}
|
2018-02-09 06:41:47 +08:00
|
|
|
setAction({G_ANYEXT, s128}, Legal);
|
2017-05-01 14:30:16 +08:00
|
|
|
|
2017-05-11 15:17:40 +08:00
|
|
|
// Comparison
|
|
|
|
setAction({G_ICMP, s1}, Legal);
|
|
|
|
|
|
|
|
for (auto Ty : {s8, s16, s32, p0})
|
|
|
|
setAction({G_ICMP, 1, Ty}, Legal);
|
2017-12-01 16:19:10 +08:00
|
|
|
|
|
|
|
// Merge/Unmerge
|
|
|
|
for (const auto &Ty : {s16, s32, s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (const auto &Ty : {s8, s16, s32}) {
|
|
|
|
setAction({G_MERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, Ty}, Legal);
|
|
|
|
}
|
2017-02-10 15:05:56 +08:00
|
|
|
}
|
|
|
|
|
2017-02-22 20:25:09 +08:00
|
|
|
void X86LegalizerInfo::setLegalizerInfo64bit() {
|
2017-02-10 15:05:56 +08:00
|
|
|
|
|
|
|
if (!Subtarget.is64Bit())
|
|
|
|
return;
|
|
|
|
|
2018-03-14 08:36:23 +08:00
|
|
|
const LLT p0 = LLT::pointer(0, TM.getPointerSizeInBits(0));
|
2018-02-28 17:18:47 +08:00
|
|
|
const LLT s1 = LLT::scalar(1);
|
|
|
|
const LLT s8 = LLT::scalar(8);
|
|
|
|
const LLT s16 = LLT::scalar(16);
|
|
|
|
const LLT s32 = LLT::scalar(32);
|
2017-02-10 15:05:56 +08:00
|
|
|
const LLT s64 = LLT::scalar(64);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT s128 = LLT::scalar(128);
|
2017-02-10 15:05:56 +08:00
|
|
|
|
2017-08-31 19:40:03 +08:00
|
|
|
setAction({G_IMPLICIT_DEF, s64}, Legal);
|
2018-02-09 06:40:31 +08:00
|
|
|
// Need to have that, as tryFoldImplicitDef will create this pattern:
|
|
|
|
// s128 = EXTEND (G_IMPLICIT_DEF s32/s64) -> s128 = G_IMPLICIT_DEF
|
|
|
|
setAction({G_IMPLICIT_DEF, s128}, Legal);
|
2017-08-24 15:06:27 +08:00
|
|
|
|
2017-09-04 17:06:45 +08:00
|
|
|
setAction({G_PHI, s64}, Legal);
|
|
|
|
|
2017-06-28 19:39:04 +08:00
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB, G_MUL, G_AND, G_OR, G_XOR})
|
2017-08-31 19:40:03 +08:00
|
|
|
setAction({BinOp, s64}, Legal);
|
2017-03-23 23:25:57 +08:00
|
|
|
|
2017-09-11 17:41:13 +08:00
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE})
|
2017-08-31 19:40:03 +08:00
|
|
|
setAction({MemOp, s64}, Legal);
|
2017-03-26 16:11:12 +08:00
|
|
|
|
|
|
|
// Pointer-handling
|
2017-05-08 17:40:43 +08:00
|
|
|
setAction({G_GEP, 1, s64}, Legal);
|
2018-02-28 17:18:47 +08:00
|
|
|
getActionDefinitionsBuilder(G_PTRTOINT)
|
|
|
|
.legalForCartesianProduct({s1, s8, s16, s32, s64}, {p0})
|
|
|
|
.maxScalar(0, s64)
|
|
|
|
.widenScalarToNextPow2(0, /*Min*/ 8);
|
2018-02-28 20:11:53 +08:00
|
|
|
getActionDefinitionsBuilder(G_INTTOPTR).legalFor({s64, p0});
|
2017-05-08 17:40:43 +08:00
|
|
|
|
2017-04-07 22:41:59 +08:00
|
|
|
// Constants
|
2017-08-31 19:40:03 +08:00
|
|
|
setAction({TargetOpcode::G_CONSTANT, s64}, Legal);
|
2017-05-01 14:30:16 +08:00
|
|
|
|
|
|
|
// Extensions
|
2017-09-11 17:41:13 +08:00
|
|
|
for (unsigned extOp : {G_ZEXT, G_SEXT, G_ANYEXT}) {
|
|
|
|
setAction({extOp, s64}, Legal);
|
|
|
|
}
|
2017-05-11 15:17:40 +08:00
|
|
|
|
|
|
|
// Comparison
|
2017-08-31 19:40:03 +08:00
|
|
|
setAction({G_ICMP, 1, s64}, Legal);
|
2017-12-01 16:19:10 +08:00
|
|
|
|
2018-03-14 23:41:11 +08:00
|
|
|
// Shifts and SDIV
|
|
|
|
getActionDefinitionsBuilder({G_SHL, G_LSHR, G_ASHR, G_SDIV})
|
[GlobalISel][X86] Support G_LSHR/G_ASHR/G_SHL
Support G_LSHR/G_ASHR/G_SHL. We have 3 variance for
shift instructions : shift gpr, shift imm, shift 1.
Currently GlobalIsel TableGen generate patterns for
shift imm and shift 1, but with shiftCount i8.
In G_LSHR/G_ASHR/G_SHL like LLVM-IR both arguments
has the same type, so for now only shift i8 can use
auto generated TableGen patterns.
The support of G_SHL/G_ASHR enables tryCombineSExt
from LegalizationArtifactCombiner.h to hit, which
results in different legalization for the following tests:
LLVM :: CodeGen/X86/GlobalISel/ext-x86-64.ll
LLVM :: CodeGen/X86/GlobalISel/gep.ll
LLVM :: CodeGen/X86/GlobalISel/legalize-ext-x86-64.mir
-; X64-NEXT: movsbl %dil, %eax
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: shll %cl, %edi
+; X64-NEXT: movl $24, %ecx
+; X64-NEXT: # kill: def $cl killed $ecx
+; X64-NEXT: sarl %cl, %edi
+; X64-NEXT: movl %edi, %eax
..which is not optimal and should be addressed later.
Rework of the patch by igorb
Reviewed By: igorb
Differential Revision: https://reviews.llvm.org/D44395
llvm-svn: 327499
2018-03-14 19:23:57 +08:00
|
|
|
.legalFor({s8, s16, s32, s64})
|
|
|
|
.clampScalar(0, s8, s64);
|
|
|
|
|
2017-12-01 16:19:10 +08:00
|
|
|
// Merge/Unmerge
|
|
|
|
setAction({G_MERGE_VALUES, s128}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, s128}, Legal);
|
|
|
|
setAction({G_MERGE_VALUES, 1, s128}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, s128}, Legal);
|
2017-03-03 16:06:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoSSE1() {
|
|
|
|
if (!Subtarget.hasSSE1())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const LLT s32 = LLT::scalar(32);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT s64 = LLT::scalar(64);
|
2017-03-03 16:06:46 +08:00
|
|
|
const LLT v4s32 = LLT::vector(4, 32);
|
2017-03-23 23:25:57 +08:00
|
|
|
const LLT v2s64 = LLT::vector(2, 64);
|
2017-03-03 16:06:46 +08:00
|
|
|
|
|
|
|
for (unsigned BinOp : {G_FADD, G_FSUB, G_FMUL, G_FDIV})
|
|
|
|
for (auto Ty : {s32, v4s32})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
2017-03-23 23:25:57 +08:00
|
|
|
|
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE})
|
|
|
|
for (auto Ty : {v4s32, v2s64})
|
|
|
|
setAction({MemOp, Ty}, Legal);
|
2017-09-17 16:08:13 +08:00
|
|
|
|
|
|
|
// Constants
|
|
|
|
setAction({TargetOpcode::G_FCONSTANT, s32}, Legal);
|
2017-12-01 16:19:10 +08:00
|
|
|
|
|
|
|
// Merge/Unmerge
|
|
|
|
for (const auto &Ty : {v4s32, v2s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
setAction({G_MERGE_VALUES, 1, s64}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, s64}, Legal);
|
2017-03-03 16:06:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoSSE2() {
|
|
|
|
if (!Subtarget.hasSSE2())
|
|
|
|
return;
|
|
|
|
|
2017-09-13 17:05:23 +08:00
|
|
|
const LLT s32 = LLT::scalar(32);
|
2017-03-03 16:06:46 +08:00
|
|
|
const LLT s64 = LLT::scalar(64);
|
2017-05-18 19:10:56 +08:00
|
|
|
const LLT v16s8 = LLT::vector(16, 8);
|
2017-05-08 17:03:37 +08:00
|
|
|
const LLT v8s16 = LLT::vector(8, 16);
|
2017-03-03 16:06:46 +08:00
|
|
|
const LLT v4s32 = LLT::vector(4, 32);
|
|
|
|
const LLT v2s64 = LLT::vector(2, 64);
|
|
|
|
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v32s8 = LLT::vector(32, 8);
|
|
|
|
const LLT v16s16 = LLT::vector(16, 16);
|
|
|
|
const LLT v8s32 = LLT::vector(8, 32);
|
|
|
|
const LLT v4s64 = LLT::vector(4, 64);
|
|
|
|
|
2017-03-03 16:06:46 +08:00
|
|
|
for (unsigned BinOp : {G_FADD, G_FSUB, G_FMUL, G_FDIV})
|
|
|
|
for (auto Ty : {s64, v2s64})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
|
|
|
|
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB})
|
2017-05-18 19:10:56 +08:00
|
|
|
for (auto Ty : {v16s8, v8s16, v4s32, v2s64})
|
2017-03-03 16:06:46 +08:00
|
|
|
setAction({BinOp, Ty}, Legal);
|
2017-05-08 17:03:37 +08:00
|
|
|
|
|
|
|
setAction({G_MUL, v8s16}, Legal);
|
2017-09-13 17:05:23 +08:00
|
|
|
|
|
|
|
setAction({G_FPEXT, s64}, Legal);
|
|
|
|
setAction({G_FPEXT, 1, s32}, Legal);
|
2017-09-17 16:08:13 +08:00
|
|
|
|
|
|
|
// Constants
|
|
|
|
setAction({TargetOpcode::G_FCONSTANT, s64}, Legal);
|
2017-12-01 16:19:10 +08:00
|
|
|
|
|
|
|
// Merge/Unmerge
|
|
|
|
for (const auto &Ty :
|
|
|
|
{v16s8, v32s8, v8s16, v16s16, v4s32, v8s32, v2s64, v4s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (const auto &Ty : {v16s8, v8s16, v4s32, v2s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, Ty}, Legal);
|
|
|
|
}
|
2017-05-08 17:03:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoSSE41() {
|
|
|
|
if (!Subtarget.hasSSE41())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const LLT v4s32 = LLT::vector(4, 32);
|
|
|
|
|
|
|
|
setAction({G_MUL, v4s32}, Legal);
|
|
|
|
}
|
|
|
|
|
2017-05-23 16:23:51 +08:00
|
|
|
void X86LegalizerInfo::setLegalizerInfoAVX() {
|
|
|
|
if (!Subtarget.hasAVX())
|
|
|
|
return;
|
|
|
|
|
2017-06-22 17:43:35 +08:00
|
|
|
const LLT v16s8 = LLT::vector(16, 8);
|
|
|
|
const LLT v8s16 = LLT::vector(8, 16);
|
|
|
|
const LLT v4s32 = LLT::vector(4, 32);
|
|
|
|
const LLT v2s64 = LLT::vector(2, 64);
|
|
|
|
|
|
|
|
const LLT v32s8 = LLT::vector(32, 8);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v64s8 = LLT::vector(64, 8);
|
2017-06-22 17:43:35 +08:00
|
|
|
const LLT v16s16 = LLT::vector(16, 16);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v32s16 = LLT::vector(32, 16);
|
2017-05-23 16:23:51 +08:00
|
|
|
const LLT v8s32 = LLT::vector(8, 32);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v16s32 = LLT::vector(16, 32);
|
2017-05-23 16:23:51 +08:00
|
|
|
const LLT v4s64 = LLT::vector(4, 64);
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v8s64 = LLT::vector(8, 64);
|
2017-05-23 16:23:51 +08:00
|
|
|
|
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE})
|
|
|
|
for (auto Ty : {v8s32, v4s64})
|
|
|
|
setAction({MemOp, Ty}, Legal);
|
2017-06-22 17:43:35 +08:00
|
|
|
|
2017-06-27 04:34:13 +08:00
|
|
|
for (auto Ty : {v32s8, v16s16, v8s32, v4s64}) {
|
2017-06-22 17:43:35 +08:00
|
|
|
setAction({G_INSERT, Ty}, Legal);
|
2017-06-27 04:34:13 +08:00
|
|
|
setAction({G_EXTRACT, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (auto Ty : {v16s8, v8s16, v4s32, v2s64}) {
|
2017-06-22 17:43:35 +08:00
|
|
|
setAction({G_INSERT, 1, Ty}, Legal);
|
2017-06-27 04:34:13 +08:00
|
|
|
setAction({G_EXTRACT, Ty}, Legal);
|
|
|
|
}
|
2017-12-01 16:19:10 +08:00
|
|
|
// Merge/Unmerge
|
|
|
|
for (const auto &Ty :
|
|
|
|
{v32s8, v64s8, v16s16, v32s16, v8s32, v16s32, v4s64, v8s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (const auto &Ty :
|
|
|
|
{v16s8, v32s8, v8s16, v16s16, v4s32, v8s32, v2s64, v4s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, Ty}, Legal);
|
|
|
|
}
|
2017-05-23 16:23:51 +08:00
|
|
|
}
|
|
|
|
|
2017-05-08 17:03:37 +08:00
|
|
|
void X86LegalizerInfo::setLegalizerInfoAVX2() {
|
|
|
|
if (!Subtarget.hasAVX2())
|
|
|
|
return;
|
|
|
|
|
2017-05-18 19:10:56 +08:00
|
|
|
const LLT v32s8 = LLT::vector(32, 8);
|
2017-05-08 17:03:37 +08:00
|
|
|
const LLT v16s16 = LLT::vector(16, 16);
|
|
|
|
const LLT v8s32 = LLT::vector(8, 32);
|
2017-05-18 19:10:56 +08:00
|
|
|
const LLT v4s64 = LLT::vector(4, 64);
|
|
|
|
|
2017-12-01 16:19:10 +08:00
|
|
|
const LLT v64s8 = LLT::vector(64, 8);
|
|
|
|
const LLT v32s16 = LLT::vector(32, 16);
|
|
|
|
const LLT v16s32 = LLT::vector(16, 32);
|
|
|
|
const LLT v8s64 = LLT::vector(8, 64);
|
|
|
|
|
2017-05-18 19:10:56 +08:00
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB})
|
|
|
|
for (auto Ty : {v32s8, v16s16, v8s32, v4s64})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
2017-05-08 17:03:37 +08:00
|
|
|
|
|
|
|
for (auto Ty : {v16s16, v8s32})
|
|
|
|
setAction({G_MUL, Ty}, Legal);
|
2017-12-01 16:19:10 +08:00
|
|
|
|
|
|
|
// Merge/Unmerge
|
|
|
|
for (const auto &Ty : {v64s8, v32s16, v16s32, v8s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (const auto &Ty : {v32s8, v16s16, v8s32, v4s64}) {
|
|
|
|
setAction({G_MERGE_VALUES, 1, Ty}, Legal);
|
|
|
|
setAction({G_UNMERGE_VALUES, Ty}, Legal);
|
|
|
|
}
|
2017-05-08 17:03:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoAVX512() {
|
|
|
|
if (!Subtarget.hasAVX512())
|
|
|
|
return;
|
|
|
|
|
2017-06-22 17:43:35 +08:00
|
|
|
const LLT v16s8 = LLT::vector(16, 8);
|
|
|
|
const LLT v8s16 = LLT::vector(8, 16);
|
|
|
|
const LLT v4s32 = LLT::vector(4, 32);
|
|
|
|
const LLT v2s64 = LLT::vector(2, 64);
|
|
|
|
|
|
|
|
const LLT v32s8 = LLT::vector(32, 8);
|
|
|
|
const LLT v16s16 = LLT::vector(16, 16);
|
|
|
|
const LLT v8s32 = LLT::vector(8, 32);
|
|
|
|
const LLT v4s64 = LLT::vector(4, 64);
|
|
|
|
|
|
|
|
const LLT v64s8 = LLT::vector(64, 8);
|
|
|
|
const LLT v32s16 = LLT::vector(32, 16);
|
2017-05-08 17:03:37 +08:00
|
|
|
const LLT v16s32 = LLT::vector(16, 32);
|
2017-05-18 19:10:56 +08:00
|
|
|
const LLT v8s64 = LLT::vector(8, 64);
|
|
|
|
|
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB})
|
|
|
|
for (auto Ty : {v16s32, v8s64})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
2017-05-08 17:03:37 +08:00
|
|
|
|
|
|
|
setAction({G_MUL, v16s32}, Legal);
|
|
|
|
|
2017-05-23 16:23:51 +08:00
|
|
|
for (unsigned MemOp : {G_LOAD, G_STORE})
|
|
|
|
for (auto Ty : {v16s32, v8s64})
|
|
|
|
setAction({MemOp, Ty}, Legal);
|
|
|
|
|
2017-06-27 04:34:13 +08:00
|
|
|
for (auto Ty : {v64s8, v32s16, v16s32, v8s64}) {
|
2017-06-22 17:43:35 +08:00
|
|
|
setAction({G_INSERT, Ty}, Legal);
|
2017-06-27 04:34:13 +08:00
|
|
|
setAction({G_EXTRACT, 1, Ty}, Legal);
|
|
|
|
}
|
|
|
|
for (auto Ty : {v32s8, v16s16, v8s32, v4s64, v16s8, v8s16, v4s32, v2s64}) {
|
2017-06-22 17:43:35 +08:00
|
|
|
setAction({G_INSERT, 1, Ty}, Legal);
|
2017-06-27 04:34:13 +08:00
|
|
|
setAction({G_EXTRACT, Ty}, Legal);
|
|
|
|
}
|
2017-06-22 17:43:35 +08:00
|
|
|
|
2017-05-08 17:03:37 +08:00
|
|
|
/************ VLX *******************/
|
|
|
|
if (!Subtarget.hasVLX())
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (auto Ty : {v4s32, v8s32})
|
|
|
|
setAction({G_MUL, Ty}, Legal);
|
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoAVX512DQ() {
|
|
|
|
if (!(Subtarget.hasAVX512() && Subtarget.hasDQI()))
|
|
|
|
return;
|
|
|
|
|
|
|
|
const LLT v8s64 = LLT::vector(8, 64);
|
|
|
|
|
|
|
|
setAction({G_MUL, v8s64}, Legal);
|
|
|
|
|
|
|
|
/************ VLX *******************/
|
|
|
|
if (!Subtarget.hasVLX())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const LLT v2s64 = LLT::vector(2, 64);
|
|
|
|
const LLT v4s64 = LLT::vector(4, 64);
|
|
|
|
|
|
|
|
for (auto Ty : {v2s64, v4s64})
|
|
|
|
setAction({G_MUL, Ty}, Legal);
|
|
|
|
}
|
|
|
|
|
|
|
|
void X86LegalizerInfo::setLegalizerInfoAVX512BW() {
|
|
|
|
if (!(Subtarget.hasAVX512() && Subtarget.hasBWI()))
|
|
|
|
return;
|
|
|
|
|
2017-05-18 19:10:56 +08:00
|
|
|
const LLT v64s8 = LLT::vector(64, 8);
|
2017-05-08 17:03:37 +08:00
|
|
|
const LLT v32s16 = LLT::vector(32, 16);
|
|
|
|
|
2017-05-18 19:10:56 +08:00
|
|
|
for (unsigned BinOp : {G_ADD, G_SUB})
|
|
|
|
for (auto Ty : {v64s8, v32s16})
|
|
|
|
setAction({BinOp, Ty}, Legal);
|
|
|
|
|
2017-05-08 17:03:37 +08:00
|
|
|
setAction({G_MUL, v32s16}, Legal);
|
|
|
|
|
|
|
|
/************ VLX *******************/
|
|
|
|
if (!Subtarget.hasVLX())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const LLT v8s16 = LLT::vector(8, 16);
|
|
|
|
const LLT v16s16 = LLT::vector(16, 16);
|
|
|
|
|
|
|
|
for (auto Ty : {v8s16, v16s16})
|
|
|
|
setAction({G_MUL, Ty}, Legal);
|
2017-02-10 15:05:56 +08:00
|
|
|
}
|