mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 02:10:53 +01:00
* Fixed exported types not being suggested in autocomplete * `T...` is now convertible to `...any` (Fixes https://github.com/Roblox/luau/issues/767) * Fixed issue with `T?` not being convertible to `T | T` or `T?` (sometimes when internal pointer identity is different) * Fixed potential crash in missing table key error suggestion to use a similar existing key * `lua_topointer` now returns a pointer for strings C++ API Changes: * `prepareModuleScope` callback has moved from TypeChecker to Frontend * For LSPs, AstQuery functions (and `isWithinComment`) can be used without full Frontend data A lot of changes in our two experimental components as well. In our work on the new type-solver, the following issues were fixed: * Fixed table union and intersection indexing * Correct custom type environments are now used * Fixed issue with values of `free & number` type not accepted in numeric operations And these are the changes in native code generation (JIT): * arm64 lowering is almost complete with support for 99% of IR commands and all fastcalls * Fixed x64 assembly encoding for extended byte registers * More external x64 calls are aware of register allocator * `math.min`/`math.max` with more than 2 arguments are now lowered to IR as well * Fixed correctness issues with `math` library calls with multiple results in variadic context and with x64 register conflicts * x64 register allocator learnt to restore values from VM memory instead of always using stack spills * x64 exception unwind information now supports multiple functions and fixes function start offset in Dwarf2 info
199 lines
7.2 KiB
C++
199 lines
7.2 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#pragma once
|
|
|
|
#include "Luau/Common.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
namespace Luau
|
|
{
|
|
namespace CodeGen
|
|
{
|
|
namespace A64
|
|
{
|
|
|
|
enum class KindA64 : uint8_t
|
|
{
|
|
none,
|
|
w, // 32-bit GPR
|
|
x, // 64-bit GPR
|
|
d, // 64-bit SIMD&FP scalar
|
|
q, // 128-bit SIMD&FP vector
|
|
};
|
|
|
|
struct RegisterA64
|
|
{
|
|
KindA64 kind : 3;
|
|
uint8_t index : 5;
|
|
|
|
constexpr bool operator==(RegisterA64 rhs) const
|
|
{
|
|
return kind == rhs.kind && index == rhs.index;
|
|
}
|
|
|
|
constexpr bool operator!=(RegisterA64 rhs) const
|
|
{
|
|
return !(*this == rhs);
|
|
}
|
|
};
|
|
|
|
constexpr RegisterA64 castReg(KindA64 kind, RegisterA64 reg)
|
|
{
|
|
LUAU_ASSERT(kind != reg.kind);
|
|
LUAU_ASSERT(kind != KindA64::none && reg.kind != KindA64::none);
|
|
LUAU_ASSERT((kind == KindA64::w || kind == KindA64::x) == (reg.kind == KindA64::w || reg.kind == KindA64::x));
|
|
|
|
return RegisterA64{kind, reg.index};
|
|
}
|
|
|
|
// This is equivalent to castReg(KindA64::x), but is separate because it implies different semantics
|
|
// Specifically, there are cases when it's useful to treat a wN register as an xN register *after* it has been assigned a value
|
|
// Since all A64 instructions that write to wN implicitly zero the top half, this works when we need zero extension semantics
|
|
// Crucially, this is *not* safe on an ABI boundary - an int parameter in wN register may have anything in its top half in certain cases
|
|
// However, as long as our codegen doesn't use 32-bit truncation by using castReg x=>w, we can safely rely on this.
|
|
constexpr RegisterA64 zextReg(RegisterA64 reg)
|
|
{
|
|
LUAU_ASSERT(reg.kind == KindA64::w);
|
|
|
|
return RegisterA64{KindA64::x, reg.index};
|
|
}
|
|
|
|
constexpr RegisterA64 noreg{KindA64::none, 0};
|
|
|
|
constexpr RegisterA64 w0{KindA64::w, 0};
|
|
constexpr RegisterA64 w1{KindA64::w, 1};
|
|
constexpr RegisterA64 w2{KindA64::w, 2};
|
|
constexpr RegisterA64 w3{KindA64::w, 3};
|
|
constexpr RegisterA64 w4{KindA64::w, 4};
|
|
constexpr RegisterA64 w5{KindA64::w, 5};
|
|
constexpr RegisterA64 w6{KindA64::w, 6};
|
|
constexpr RegisterA64 w7{KindA64::w, 7};
|
|
constexpr RegisterA64 w8{KindA64::w, 8};
|
|
constexpr RegisterA64 w9{KindA64::w, 9};
|
|
constexpr RegisterA64 w10{KindA64::w, 10};
|
|
constexpr RegisterA64 w11{KindA64::w, 11};
|
|
constexpr RegisterA64 w12{KindA64::w, 12};
|
|
constexpr RegisterA64 w13{KindA64::w, 13};
|
|
constexpr RegisterA64 w14{KindA64::w, 14};
|
|
constexpr RegisterA64 w15{KindA64::w, 15};
|
|
constexpr RegisterA64 w16{KindA64::w, 16};
|
|
constexpr RegisterA64 w17{KindA64::w, 17};
|
|
constexpr RegisterA64 w18{KindA64::w, 18};
|
|
constexpr RegisterA64 w19{KindA64::w, 19};
|
|
constexpr RegisterA64 w20{KindA64::w, 20};
|
|
constexpr RegisterA64 w21{KindA64::w, 21};
|
|
constexpr RegisterA64 w22{KindA64::w, 22};
|
|
constexpr RegisterA64 w23{KindA64::w, 23};
|
|
constexpr RegisterA64 w24{KindA64::w, 24};
|
|
constexpr RegisterA64 w25{KindA64::w, 25};
|
|
constexpr RegisterA64 w26{KindA64::w, 26};
|
|
constexpr RegisterA64 w27{KindA64::w, 27};
|
|
constexpr RegisterA64 w28{KindA64::w, 28};
|
|
constexpr RegisterA64 w29{KindA64::w, 29};
|
|
constexpr RegisterA64 w30{KindA64::w, 30};
|
|
constexpr RegisterA64 wzr{KindA64::w, 31};
|
|
|
|
constexpr RegisterA64 x0{KindA64::x, 0};
|
|
constexpr RegisterA64 x1{KindA64::x, 1};
|
|
constexpr RegisterA64 x2{KindA64::x, 2};
|
|
constexpr RegisterA64 x3{KindA64::x, 3};
|
|
constexpr RegisterA64 x4{KindA64::x, 4};
|
|
constexpr RegisterA64 x5{KindA64::x, 5};
|
|
constexpr RegisterA64 x6{KindA64::x, 6};
|
|
constexpr RegisterA64 x7{KindA64::x, 7};
|
|
constexpr RegisterA64 x8{KindA64::x, 8};
|
|
constexpr RegisterA64 x9{KindA64::x, 9};
|
|
constexpr RegisterA64 x10{KindA64::x, 10};
|
|
constexpr RegisterA64 x11{KindA64::x, 11};
|
|
constexpr RegisterA64 x12{KindA64::x, 12};
|
|
constexpr RegisterA64 x13{KindA64::x, 13};
|
|
constexpr RegisterA64 x14{KindA64::x, 14};
|
|
constexpr RegisterA64 x15{KindA64::x, 15};
|
|
constexpr RegisterA64 x16{KindA64::x, 16};
|
|
constexpr RegisterA64 x17{KindA64::x, 17};
|
|
constexpr RegisterA64 x18{KindA64::x, 18};
|
|
constexpr RegisterA64 x19{KindA64::x, 19};
|
|
constexpr RegisterA64 x20{KindA64::x, 20};
|
|
constexpr RegisterA64 x21{KindA64::x, 21};
|
|
constexpr RegisterA64 x22{KindA64::x, 22};
|
|
constexpr RegisterA64 x23{KindA64::x, 23};
|
|
constexpr RegisterA64 x24{KindA64::x, 24};
|
|
constexpr RegisterA64 x25{KindA64::x, 25};
|
|
constexpr RegisterA64 x26{KindA64::x, 26};
|
|
constexpr RegisterA64 x27{KindA64::x, 27};
|
|
constexpr RegisterA64 x28{KindA64::x, 28};
|
|
constexpr RegisterA64 x29{KindA64::x, 29};
|
|
constexpr RegisterA64 x30{KindA64::x, 30};
|
|
constexpr RegisterA64 xzr{KindA64::x, 31};
|
|
|
|
constexpr RegisterA64 sp{KindA64::none, 31};
|
|
|
|
constexpr RegisterA64 d0{KindA64::d, 0};
|
|
constexpr RegisterA64 d1{KindA64::d, 1};
|
|
constexpr RegisterA64 d2{KindA64::d, 2};
|
|
constexpr RegisterA64 d3{KindA64::d, 3};
|
|
constexpr RegisterA64 d4{KindA64::d, 4};
|
|
constexpr RegisterA64 d5{KindA64::d, 5};
|
|
constexpr RegisterA64 d6{KindA64::d, 6};
|
|
constexpr RegisterA64 d7{KindA64::d, 7};
|
|
constexpr RegisterA64 d8{KindA64::d, 8};
|
|
constexpr RegisterA64 d9{KindA64::d, 9};
|
|
constexpr RegisterA64 d10{KindA64::d, 10};
|
|
constexpr RegisterA64 d11{KindA64::d, 11};
|
|
constexpr RegisterA64 d12{KindA64::d, 12};
|
|
constexpr RegisterA64 d13{KindA64::d, 13};
|
|
constexpr RegisterA64 d14{KindA64::d, 14};
|
|
constexpr RegisterA64 d15{KindA64::d, 15};
|
|
constexpr RegisterA64 d16{KindA64::d, 16};
|
|
constexpr RegisterA64 d17{KindA64::d, 17};
|
|
constexpr RegisterA64 d18{KindA64::d, 18};
|
|
constexpr RegisterA64 d19{KindA64::d, 19};
|
|
constexpr RegisterA64 d20{KindA64::d, 20};
|
|
constexpr RegisterA64 d21{KindA64::d, 21};
|
|
constexpr RegisterA64 d22{KindA64::d, 22};
|
|
constexpr RegisterA64 d23{KindA64::d, 23};
|
|
constexpr RegisterA64 d24{KindA64::d, 24};
|
|
constexpr RegisterA64 d25{KindA64::d, 25};
|
|
constexpr RegisterA64 d26{KindA64::d, 26};
|
|
constexpr RegisterA64 d27{KindA64::d, 27};
|
|
constexpr RegisterA64 d28{KindA64::d, 28};
|
|
constexpr RegisterA64 d29{KindA64::d, 29};
|
|
constexpr RegisterA64 d30{KindA64::d, 30};
|
|
constexpr RegisterA64 d31{KindA64::d, 31};
|
|
|
|
constexpr RegisterA64 q0{KindA64::q, 0};
|
|
constexpr RegisterA64 q1{KindA64::q, 1};
|
|
constexpr RegisterA64 q2{KindA64::q, 2};
|
|
constexpr RegisterA64 q3{KindA64::q, 3};
|
|
constexpr RegisterA64 q4{KindA64::q, 4};
|
|
constexpr RegisterA64 q5{KindA64::q, 5};
|
|
constexpr RegisterA64 q6{KindA64::q, 6};
|
|
constexpr RegisterA64 q7{KindA64::q, 7};
|
|
constexpr RegisterA64 q8{KindA64::q, 8};
|
|
constexpr RegisterA64 q9{KindA64::q, 9};
|
|
constexpr RegisterA64 q10{KindA64::q, 10};
|
|
constexpr RegisterA64 q11{KindA64::q, 11};
|
|
constexpr RegisterA64 q12{KindA64::q, 12};
|
|
constexpr RegisterA64 q13{KindA64::q, 13};
|
|
constexpr RegisterA64 q14{KindA64::q, 14};
|
|
constexpr RegisterA64 q15{KindA64::q, 15};
|
|
constexpr RegisterA64 q16{KindA64::q, 16};
|
|
constexpr RegisterA64 q17{KindA64::q, 17};
|
|
constexpr RegisterA64 q18{KindA64::q, 18};
|
|
constexpr RegisterA64 q19{KindA64::q, 19};
|
|
constexpr RegisterA64 q20{KindA64::q, 20};
|
|
constexpr RegisterA64 q21{KindA64::q, 21};
|
|
constexpr RegisterA64 q22{KindA64::q, 22};
|
|
constexpr RegisterA64 q23{KindA64::q, 23};
|
|
constexpr RegisterA64 q24{KindA64::q, 24};
|
|
constexpr RegisterA64 q25{KindA64::q, 25};
|
|
constexpr RegisterA64 q26{KindA64::q, 26};
|
|
constexpr RegisterA64 q27{KindA64::q, 27};
|
|
constexpr RegisterA64 q28{KindA64::q, 28};
|
|
constexpr RegisterA64 q29{KindA64::q, 29};
|
|
constexpr RegisterA64 q30{KindA64::q, 30};
|
|
constexpr RegisterA64 q31{KindA64::q, 31};
|
|
|
|
} // namespace A64
|
|
} // namespace CodeGen
|
|
} // namespace Luau
|