mirror of
https://github.com/luau-lang/luau.git
synced 2025-08-26 11:27:08 +01:00
# General * Expose an optional `get_alias` API as an alternative to `get_config` in Luau.Require and Luau.RequireNavigator. * Improve the Luau CLI's virtual filesystem implementation to fix bugs related to `init.luau`. Fixes https://github.com/luau-lang/luau/issues/1816 # New Type Solver * Avoid double reporting errors when erroneous arguments are provided to type functions. * Fix some instances of unresovable cyclic type functions in loops by only considering the first loop cycles. This results in some type inference inaccuracies when the type of a variable in loop through multiple iterations. Fixes https://github.com/luau-lang/luau/issues/1413. * Better generalize free types that have meaningful lower and upper bounds, especially for table indexers. * Report more specific errors when assigning or returning table literal types, instead of citing the *entire* table type. * Inference for functions with generic type packs is greatly improved. * Fix some internal compiler exceptions when using type-stating functions like `table.freeze` in `if _ then _ else _` expressions and short circuiting binary operations. * More consistently simplify unions of primitive types, especially in array-like and dictionary-like tables. * Fix a crash when type checking an erroneous type alias containing `typeof` with a type assertion expression, as in: ``` type MyTable = {} -- This will error at type checking time as it's a duplicate type MyTable = typeof(setmetatable(SomeTable :: {}, SomeMetaTable)); ``` * Fix a crash when inferring the type of an index expression where the indexee is invalid (e.g. `nil`). # Runtime * Avoid throwing an exception from `luau_load` if we run out of memory. * Type functions are no longer compiled and included in bytecode. Fixes #1817. * Fix some instances of Luau C API functions reading invalid debug information (generally when the first or last instruction of a block was being inspected). Fixes #1369. * Avoid potential signed integer overflow when doing bounds checks on tables. * Support 16 byte aligned userdata objects when system allocation alignment is also 16 bytes. * Fix memory leaks in `Luau.Require` when using VM build with no exceptions. Fixes #1827. --------- Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: James McNellis <jmcnellis@roblox.com> Co-authored-by: Sora Kanosue <skanosue@roblox.com> Co-authored-by: Talha Pathan <tpathan@roblox.com> Co-authored-by: Varun Saini <vsaini@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
145 lines
3.7 KiB
C++
145 lines
3.7 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/CodeGenCommon.h"
|
|
#include "Luau/RegisterX64.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
namespace Luau
|
|
{
|
|
namespace CodeGen
|
|
{
|
|
namespace X64
|
|
{
|
|
|
|
enum class CategoryX64 : uint8_t
|
|
{
|
|
reg,
|
|
mem,
|
|
imm,
|
|
};
|
|
|
|
struct OperandX64
|
|
{
|
|
constexpr OperandX64(RegisterX64 reg)
|
|
: cat(CategoryX64::reg)
|
|
, index(noreg)
|
|
, base(reg)
|
|
, memSize(SizeX64::none)
|
|
, scale(1)
|
|
, imm(0)
|
|
{
|
|
}
|
|
|
|
constexpr OperandX64(int32_t imm)
|
|
: cat(CategoryX64::imm)
|
|
, index(noreg)
|
|
, base(noreg)
|
|
, memSize(SizeX64::none)
|
|
, scale(1)
|
|
, imm(imm)
|
|
{
|
|
}
|
|
|
|
constexpr explicit OperandX64(SizeX64 size, RegisterX64 index, uint8_t scale, RegisterX64 base, int32_t disp)
|
|
: cat(CategoryX64::mem)
|
|
, index(index)
|
|
, base(base)
|
|
, memSize(size)
|
|
, scale(scale)
|
|
, imm(disp)
|
|
{
|
|
}
|
|
|
|
// Fields are carefully placed to make this struct fit into an 8 byte register
|
|
CategoryX64 cat;
|
|
RegisterX64 index;
|
|
RegisterX64 base;
|
|
SizeX64 memSize : 4;
|
|
uint8_t scale : 4;
|
|
int32_t imm;
|
|
|
|
constexpr OperandX64 operator[](OperandX64&& addr) const
|
|
{
|
|
CODEGEN_ASSERT(cat == CategoryX64::mem);
|
|
CODEGEN_ASSERT(index == noreg && scale == 1 && base == noreg && imm == 0);
|
|
CODEGEN_ASSERT(addr.memSize == SizeX64::none);
|
|
|
|
addr.cat = CategoryX64::mem;
|
|
addr.memSize = memSize;
|
|
return addr;
|
|
}
|
|
};
|
|
|
|
inline constexpr OperandX64 addr{SizeX64::none, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 byte{SizeX64::byte, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 word{SizeX64::word, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 dword{SizeX64::dword, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 qword{SizeX64::qword, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 xmmword{SizeX64::xmmword, noreg, 1, noreg, 0};
|
|
inline constexpr OperandX64 ymmword{SizeX64::ymmword, noreg, 1, noreg, 0};
|
|
|
|
constexpr OperandX64 operator*(RegisterX64 reg, uint8_t scale)
|
|
{
|
|
if (scale == 1)
|
|
return OperandX64(reg);
|
|
|
|
CODEGEN_ASSERT(scale == 1 || scale == 2 || scale == 4 || scale == 8);
|
|
CODEGEN_ASSERT(reg.index != 0b100 && "can't scale SP");
|
|
|
|
return OperandX64(SizeX64::none, reg, scale, noreg, 0);
|
|
}
|
|
|
|
constexpr OperandX64 operator+(RegisterX64 reg, int32_t disp)
|
|
{
|
|
return OperandX64(SizeX64::none, noreg, 1, reg, disp);
|
|
}
|
|
|
|
constexpr OperandX64 operator-(RegisterX64 reg, int32_t disp)
|
|
{
|
|
return OperandX64(SizeX64::none, noreg, 1, reg, -disp);
|
|
}
|
|
|
|
constexpr OperandX64 operator+(RegisterX64 base, RegisterX64 index)
|
|
{
|
|
CODEGEN_ASSERT(index.index != 4 && "sp cannot be used as index");
|
|
CODEGEN_ASSERT(base.size == index.size);
|
|
|
|
return OperandX64(SizeX64::none, index, 1, base, 0);
|
|
}
|
|
|
|
constexpr OperandX64 operator+(OperandX64 op, int32_t disp)
|
|
{
|
|
CODEGEN_ASSERT(op.cat == CategoryX64::mem);
|
|
CODEGEN_ASSERT(op.memSize == SizeX64::none);
|
|
|
|
op.imm += disp;
|
|
return op;
|
|
}
|
|
|
|
constexpr OperandX64 operator+(OperandX64 op, RegisterX64 base)
|
|
{
|
|
CODEGEN_ASSERT(op.cat == CategoryX64::mem);
|
|
CODEGEN_ASSERT(op.memSize == SizeX64::none);
|
|
CODEGEN_ASSERT(op.base == noreg);
|
|
CODEGEN_ASSERT(op.index == noreg || op.index.size == base.size);
|
|
|
|
op.base = base;
|
|
return op;
|
|
}
|
|
|
|
constexpr OperandX64 operator+(RegisterX64 base, OperandX64 op)
|
|
{
|
|
CODEGEN_ASSERT(op.cat == CategoryX64::mem);
|
|
CODEGEN_ASSERT(op.memSize == SizeX64::none);
|
|
CODEGEN_ASSERT(op.base == noreg);
|
|
CODEGEN_ASSERT(op.index == noreg || op.index.size == base.size);
|
|
|
|
op.base = base;
|
|
return op;
|
|
}
|
|
|
|
} // namespace X64
|
|
} // namespace CodeGen
|
|
} // namespace Luau
|