1
+ local Util = require (" lazy.core.util" )
2
+
1
3
local M = {}
2
4
3
5
--- @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 ())
6
10
7
- M .TIMER = 10
8
- M .BUDGET = 100
11
+ M .BUDGET = 10
9
12
10
13
--- @type table<thread , Async>
11
14
M ._threads = setmetatable ({}, { __mode = " k" })
@@ -42,11 +45,6 @@ function Async:init(fn)
42
45
return M .add (self )
43
46
end
44
47
45
- function Async :restart ()
46
- assert (not self :running (), " Cannot restart a running async" )
47
- self :init (self ._fn )
48
- end
49
-
50
48
--- @param event AsyncEvent
51
49
--- @param cb async fun ( res : any , async : Async )
52
50
function Async :on (event , cb )
@@ -77,27 +75,41 @@ function Async:sleep(ms)
77
75
end
78
76
79
77
--- @async
80
- function Async :suspend ()
78
+ --- @param yield ? boolean
79
+ function Async :suspend (yield )
81
80
self ._suspended = true
82
- if coroutine.running () == self ._co then
81
+ if coroutine.running () == self ._co and yield ~= false then
83
82
coroutine.yield ()
84
83
end
85
84
end
86
85
87
86
function Async :resume ()
88
87
self ._suspended = false
88
+ M ._run ()
89
89
end
90
90
91
- function Async :wait ()
91
+ --- @async
92
+ --- @param yield ? boolean
93
+ function Async :wake (yield )
92
94
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 ()
93
104
if coroutine.running () == self ._co then
94
105
error (" Cannot wait on self" )
95
106
end
96
107
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
101
113
vim .wait (10 )
102
114
end
103
115
end
@@ -121,33 +133,42 @@ function Async:step()
121
133
end
122
134
123
135
function M .step ()
124
- local budget = M .BUDGET * 1e6
125
136
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
136
139
break
137
140
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
138
149
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
140
157
return M ._executor :stop ()
141
158
end
142
159
end
143
160
144
161
--- @param async Async
145
162
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 ()
147
169
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 ))
149
171
end
150
- return async
151
172
end
152
173
153
174
function M .running ()
0 commit comments