Skip to content

Commit c7ed87f

Browse files
authored
perf: automatically suspend the scheduler when all threads are waiting (#1591)
* perf: automatically suspend the scheduler when all threads are waiting * ci: fix ci * test: cleanup
1 parent 0507e19 commit c7ed87f

File tree

9 files changed

+80
-53
lines changed

9 files changed

+80
-53
lines changed

.github/workflows/ci.yml

+3-1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ jobs:
2828
./tests/run
2929
docs:
3030
runs-on: ubuntu-latest
31+
if: ${{ github.ref == 'refs/heads/main' && github.repository_owner == 'folke' }}
3132
needs: tests
3233
env:
3334
GH_TOKEN: ${{ github.token }}
@@ -40,6 +41,7 @@ jobs:
4041
run: gh workflow run "Deploy to Github Pages" --ref docs
4142
community:
4243
runs-on: ubuntu-latest
44+
if: ${{ github.ref == 'refs/heads/main' && github.repository_owner == 'folke' }}
4345
steps:
4446
- uses: actions/checkout@v4
4547
- name: Install Neovim
@@ -70,7 +72,7 @@ jobs:
7072
commit_author: "github-actions[bot] <github-actions[bot]@users.noreply.github.com>"
7173
release:
7274
name: release
73-
if: ${{ github.ref == 'refs/heads/main' }}
75+
if: ${{ github.ref == 'refs/heads/main' && github.repository_owner == 'folke' }}
7476
needs:
7577
- tests
7678
- docs

lua/lazy/async.lua

+52-31
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,14 @@
1+
local Util = require("lazy.core.util")
2+
13
local M = {}
24

35
---@type Async[]
4-
M._queue = {}
5-
M._executor = assert(vim.loop.new_timer())
6+
M._active = {}
7+
---@type Async[]
8+
M._suspended = {}
9+
M._executor = assert(vim.loop.new_check())
610

7-
M.TIMER = 10
8-
M.BUDGET = 100
11+
M.BUDGET = 10
912

1013
---@type table<thread, Async>
1114
M._threads = setmetatable({}, { __mode = "k" })
@@ -42,11 +45,6 @@ function Async:init(fn)
4245
return M.add(self)
4346
end
4447

45-
function Async:restart()
46-
assert(not self:running(), "Cannot restart a running async")
47-
self:init(self._fn)
48-
end
49-
5048
---@param event AsyncEvent
5149
---@param cb async fun(res:any, async:Async)
5250
function Async:on(event, cb)
@@ -77,27 +75,41 @@ function Async:sleep(ms)
7775
end
7876

7977
---@async
80-
function Async:suspend()
78+
---@param yield? boolean
79+
function Async:suspend(yield)
8180
self._suspended = true
82-
if coroutine.running() == self._co then
81+
if coroutine.running() == self._co and yield ~= false then
8382
coroutine.yield()
8483
end
8584
end
8685

8786
function Async:resume()
8887
self._suspended = false
88+
M._run()
8989
end
9090

91-
function Async:wait()
91+
---@async
92+
---@param yield? boolean
93+
function Async:wake(yield)
9294
local async = M.running()
95+
assert(async, "Not in an async context")
96+
self:on("done", function()
97+
async:resume()
98+
end)
99+
async:suspend(yield)
100+
end
101+
102+
---@async
103+
function Async:wait()
93104
if coroutine.running() == self._co then
94105
error("Cannot wait on self")
95106
end
96107

97-
while self:running() do
98-
if async then
99-
coroutine.yield()
100-
else
108+
local async = M.running()
109+
if async then
110+
self:wake()
111+
else
112+
while self:running() do
101113
vim.wait(10)
102114
end
103115
end
@@ -121,33 +133,42 @@ function Async:step()
121133
end
122134

123135
function M.step()
124-
local budget = M.BUDGET * 1e6
125136
local start = vim.uv.hrtime()
126-
local count = #M._queue
127-
local i = 0
128-
while #M._queue > 0 and vim.uv.hrtime() - start < budget do
129-
---@type Async
130-
local state = table.remove(M._queue, 1)
131-
if state:step() then
132-
table.insert(M._queue, state)
133-
end
134-
i = i + 1
135-
if i >= count then
137+
for _ = 1, #M._active do
138+
if vim.uv.hrtime() - start > M.BUDGET * 1e6 then
136139
break
137140
end
141+
local state = table.remove(M._active, 1)
142+
if state:step() then
143+
if state._suspended then
144+
table.insert(M._suspended, state)
145+
else
146+
table.insert(M._active, state)
147+
end
148+
end
138149
end
139-
if #M._queue == 0 then
150+
for _ = 1, #M._suspended do
151+
local state = table.remove(M._suspended, 1)
152+
table.insert(state._suspended and M._suspended or M._active, state)
153+
end
154+
155+
-- print("step", #M._active, #M._suspended)
156+
if #M._active == 0 then
140157
return M._executor:stop()
141158
end
142159
end
143160

144161
---@param async Async
145162
function M.add(async)
146-
table.insert(M._queue, async)
163+
table.insert(M._active, async)
164+
M._run()
165+
return async
166+
end
167+
168+
function M._run()
147169
if not M._executor:is_active() then
148-
M._executor:start(1, M.TIMER, vim.schedule_wrap(M.step))
170+
M._executor:start(vim.schedule_wrap(M.step))
149171
end
150-
return async
151172
end
152173

153174
function M.running()

lua/lazy/manage/runner.lua

+24-15
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,7 @@ function Runner:_start()
7878
---@type number?
7979
local wait_step = nil
8080

81+
---@async
8182
---@param resume? boolean
8283
local function continue(resume)
8384
active = 0
@@ -114,22 +115,30 @@ function Runner:_start()
114115
end
115116
local s = state[name]
116117
local plugin = self:plugin(name)
117-
if s.step == #self._pipeline then
118-
-- done
119-
s.task = nil
120-
plugin._.working = false
121-
elseif s.step < #self._pipeline then
122-
-- next
123-
s.step = s.step + 1
124-
local step = self._pipeline[s.step]
125-
if step.task == "wait" then
118+
while s.step <= #self._pipeline do
119+
if s.step == #self._pipeline then
120+
-- done
121+
s.task = nil
126122
plugin._.working = false
127-
waiting = waiting + 1
128-
wait_step = s.step
129-
else
130-
s.task = self:queue(plugin, step)
131-
plugin._.working = true
132-
active = active + 1
123+
break
124+
elseif s.step < #self._pipeline then
125+
-- next
126+
s.step = s.step + 1
127+
local step = self._pipeline[s.step]
128+
if step.task == "wait" then
129+
plugin._.working = false
130+
waiting = waiting + 1
131+
wait_step = s.step
132+
break
133+
else
134+
s.task = self:queue(plugin, step)
135+
plugin._.working = true
136+
if s.task then
137+
active = active + 1
138+
s.task:wake(false)
139+
break
140+
end
141+
end
133142
end
134143
end
135144
end

lua/lazy/manage/task/git.lua

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ M.log = {
2121
---@async
2222
---@param opts {args?: string[], updated?:boolean, check?:boolean}
2323
run = function(self, opts)
24+
-- self:spawn({ "sleep", "5" })
2425
local args = {
2526
"log",
2627
"--pretty=format:%h %s (%cr)",

tests/core/init_spec.lua

-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
---@module 'luassert'
21
local Util = require("lazy.core.util")
32

43
describe("init", function()

tests/core/plugin_spec.lua

-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
---@module 'luassert'
2-
31
local Config = require("lazy.core.config")
42
local Handler = require("lazy.core.handler")
53
local Plugin = require("lazy.core.plugin")

tests/handlers/keys_spec.lua

-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
---@module 'luassert'
21
local Keys = require("lazy.core.handler.keys")
32

43
describe("keys", function()

tests/manage/process_spec.lua

-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
---@module 'luassert'
21
local Async = require("lazy.async")
32
local Process = require("lazy.manage.process")
43

tests/manage/task_spec.lua

-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
---@module 'luassert'
21
--# selene:allow(incorrect_standard_library_use)
32
local Task = require("lazy.manage.task")
43

0 commit comments

Comments
 (0)