mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-05 19:09:11 +00:00
02241b6d24
In this update, we continue to improve the overall stability of the new type solver. We're also shipping some early bits of two new features, one of the language and one of the analysis API: user-defined type functions and an incremental typechecking API. If you use the new solver and want to use all new fixes included in this release, you have to reference an additional Luau flag: ```c++ LUAU_DYNAMIC_FASTINT(LuauTypeSolverRelease) ``` And set its value to `645`: ```c++ DFInt::LuauTypeSolverRelease.value = 645; // Or a higher value for future updates ``` ## New Solver * Fix a crash where scopes are incorrectly accessed cross-module after they've been deallocated by appropriately zeroing out associated scope pointers for free types, generic types, table types, etc. * Fix a crash where we were incorrectly caching results for bound types in generalization. * Eliminated some unnecessary intermediate allocations in the constraint solver and type function infrastructure. * Built some initial groundwork for an incremental typecheck API for use by language servers. * Built an initial technical preview for [user-defined type functions](https://rfcs.luau-lang.org/user-defined-type-functions.html), more work still to come (including calling type functions from other type functions), but adventurous folks wanting to experiment with it can try it out by enabling `FFlag::LuauUserDefinedTypeFunctionsSyntax` and `FFlag::LuauUserDefinedTypeFunction` in their local environment. Special thanks to @joonyoo181 who built up all the initial infrastructure for this during his internship! ## Miscellaneous changes * Fix a compilation error on Ubuntu (fixes #1437) --- Internal Contributors: Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Jeremy Yoo <jyoo@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> --------- Co-authored-by: Alexander McCord <amccord@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Vighnesh <vvijay@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: David Cope <dcope@roblox.com> Co-authored-by: Lily Brown <lbrown@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> Co-authored-by: Junseo Yoo <jyoo@roblox.com>
240 lines
5.5 KiB
C++
240 lines
5.5 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 <vector>
|
|
#include <memory>
|
|
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
|
|
LUAU_FASTFLAG(DebugLuauTimeTracing)
|
|
|
|
namespace Luau
|
|
{
|
|
namespace TimeTrace
|
|
{
|
|
double getClock();
|
|
uint32_t getClockMicroseconds();
|
|
} // namespace TimeTrace
|
|
} // namespace Luau
|
|
|
|
#if defined(LUAU_ENABLE_TIME_TRACE)
|
|
|
|
namespace Luau
|
|
{
|
|
namespace TimeTrace
|
|
{
|
|
struct Token
|
|
{
|
|
const char* name;
|
|
const char* category;
|
|
};
|
|
|
|
enum class EventType : uint8_t
|
|
{
|
|
Enter,
|
|
Leave,
|
|
|
|
ArgName,
|
|
ArgValue,
|
|
};
|
|
|
|
struct Event
|
|
{
|
|
EventType type;
|
|
uint16_t token;
|
|
|
|
union
|
|
{
|
|
uint32_t microsec; // 1 hour trace limit
|
|
uint32_t dataPos;
|
|
} data;
|
|
};
|
|
|
|
struct GlobalContext;
|
|
struct ThreadContext;
|
|
|
|
std::shared_ptr<GlobalContext> getGlobalContext();
|
|
|
|
uint16_t createToken(GlobalContext& context, const char* name, const char* category);
|
|
uint32_t createThread(GlobalContext& context, ThreadContext* threadContext);
|
|
void releaseThread(GlobalContext& context, ThreadContext* threadContext);
|
|
void flushEvents(GlobalContext& context, uint32_t threadId, const std::vector<Event>& events, const std::vector<char>& data);
|
|
|
|
struct ThreadContext
|
|
{
|
|
ThreadContext()
|
|
: globalContext(getGlobalContext())
|
|
{
|
|
threadId = createThread(*globalContext, this);
|
|
}
|
|
|
|
~ThreadContext()
|
|
{
|
|
if (!events.empty())
|
|
flushEvents();
|
|
|
|
releaseThread(*globalContext, this);
|
|
}
|
|
|
|
void flushEvents()
|
|
{
|
|
static uint16_t flushToken = createToken(*globalContext, "flushEvents", "TimeTrace");
|
|
|
|
events.push_back({EventType::Enter, flushToken, {getClockMicroseconds()}});
|
|
|
|
TimeTrace::flushEvents(*globalContext, threadId, events, data);
|
|
|
|
events.clear();
|
|
data.clear();
|
|
|
|
events.push_back({EventType::Leave, 0, {getClockMicroseconds()}});
|
|
}
|
|
|
|
void eventEnter(uint16_t token)
|
|
{
|
|
eventEnter(token, getClockMicroseconds());
|
|
}
|
|
|
|
void eventEnter(uint16_t token, uint32_t microsec)
|
|
{
|
|
events.push_back({EventType::Enter, token, {microsec}});
|
|
}
|
|
|
|
void eventLeave()
|
|
{
|
|
eventLeave(getClockMicroseconds());
|
|
}
|
|
|
|
void eventLeave(uint32_t microsec)
|
|
{
|
|
events.push_back({EventType::Leave, 0, {microsec}});
|
|
|
|
if (events.size() > kEventFlushLimit)
|
|
flushEvents();
|
|
}
|
|
|
|
void eventArgument(const char* name, const char* value)
|
|
{
|
|
uint32_t pos = uint32_t(data.size());
|
|
data.insert(data.end(), name, name + strlen(name) + 1);
|
|
events.push_back({EventType::ArgName, 0, {pos}});
|
|
|
|
pos = uint32_t(data.size());
|
|
data.insert(data.end(), value, value + strlen(value) + 1);
|
|
events.push_back({EventType::ArgValue, 0, {pos}});
|
|
}
|
|
|
|
std::shared_ptr<GlobalContext> globalContext;
|
|
uint32_t threadId;
|
|
std::vector<Event> events;
|
|
std::vector<char> data;
|
|
|
|
static constexpr size_t kEventFlushLimit = 8192;
|
|
};
|
|
|
|
using ThreadContextProvider = ThreadContext& (*)();
|
|
|
|
inline ThreadContextProvider& threadContextProvider()
|
|
{
|
|
static ThreadContextProvider handler = nullptr;
|
|
return handler;
|
|
}
|
|
|
|
ThreadContext& getThreadContext();
|
|
|
|
struct Scope
|
|
{
|
|
explicit Scope(uint16_t token)
|
|
: context(getThreadContext())
|
|
{
|
|
if (!FFlag::DebugLuauTimeTracing)
|
|
return;
|
|
|
|
context.eventEnter(token);
|
|
}
|
|
|
|
~Scope()
|
|
{
|
|
if (!FFlag::DebugLuauTimeTracing)
|
|
return;
|
|
|
|
context.eventLeave();
|
|
}
|
|
|
|
ThreadContext& context;
|
|
};
|
|
|
|
struct OptionalTailScope
|
|
{
|
|
explicit OptionalTailScope(uint16_t token, uint32_t threshold)
|
|
: context(getThreadContext())
|
|
, token(token)
|
|
, threshold(threshold)
|
|
{
|
|
if (!FFlag::DebugLuauTimeTracing)
|
|
return;
|
|
|
|
pos = uint32_t(context.events.size());
|
|
microsec = getClockMicroseconds();
|
|
}
|
|
|
|
~OptionalTailScope()
|
|
{
|
|
if (!FFlag::DebugLuauTimeTracing)
|
|
return;
|
|
|
|
if (pos == context.events.size())
|
|
{
|
|
uint32_t curr = getClockMicroseconds();
|
|
|
|
if (curr - microsec > threshold)
|
|
{
|
|
context.eventEnter(token, microsec);
|
|
context.eventLeave(curr);
|
|
}
|
|
}
|
|
}
|
|
|
|
ThreadContext& context;
|
|
uint16_t token;
|
|
uint32_t threshold;
|
|
uint32_t microsec;
|
|
uint32_t pos;
|
|
};
|
|
|
|
LUAU_NOINLINE uint16_t createScopeData(const char* name, const char* category);
|
|
|
|
} // namespace TimeTrace
|
|
} // namespace Luau
|
|
|
|
// Regular scope
|
|
#define LUAU_TIMETRACE_SCOPE(name, category) \
|
|
static uint16_t lttScopeStatic = Luau::TimeTrace::createScopeData(name, category); \
|
|
Luau::TimeTrace::Scope lttScope(lttScopeStatic)
|
|
|
|
// A scope without nested scopes that may be skipped if the time it took is less than the threshold
|
|
#define LUAU_TIMETRACE_OPTIONAL_TAIL_SCOPE(name, category, microsec) \
|
|
static uint16_t lttScopeStaticOptTail = Luau::TimeTrace::createScopeData(name, category); \
|
|
Luau::TimeTrace::OptionalTailScope lttScope(lttScopeStaticOptTail, microsec)
|
|
|
|
// Extra key/value data can be added to regular scopes
|
|
#define LUAU_TIMETRACE_ARGUMENT(name, value) \
|
|
do \
|
|
{ \
|
|
if (FFlag::DebugLuauTimeTracing) \
|
|
lttScope.context.eventArgument(name, value); \
|
|
} while (false)
|
|
|
|
#else
|
|
|
|
#define LUAU_TIMETRACE_SCOPE(name, category)
|
|
#define LUAU_TIMETRACE_OPTIONAL_TAIL_SCOPE(name, category, microsec)
|
|
#define LUAU_TIMETRACE_ARGUMENT(name, value) \
|
|
do \
|
|
{ \
|
|
} while (false)
|
|
|
|
#endif
|