Files
llgo/x/async/async_llgo.go

129 lines
2.9 KiB
Go
Raw Normal View History

//go:build llgo
// +build llgo
2024-09-04 17:08:08 +08:00
/*
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package async
import (
"sync"
2024-09-04 17:08:08 +08:00
"sync/atomic"
"github.com/goplus/llgo/c/libuv"
"github.com/goplus/llgo/x/cbind"
2024-09-04 17:08:08 +08:00
)
// Currently Async run chain a future that call chain in the goroutine running `async.Run`.
// Basic implementation:
// func Async[T any](fn func(func(T))) Future[T] {
// return func(chain func(T)) {
// loop := Exec().L
// var result T
// var a *libuv.Async
// var cb libuv.AsyncCb
// a, cb = cbind.BindF[libuv.Async, libuv.AsyncCb](func(a *libuv.Async) {
// a.Close(nil)
// chain(result)
// })
// loop.Async(a, cb)
// fn(func(v T) {
// result = v
// a.Send()
// })
// }
// }
//
// Better implementation to support multiple callbacks:
func Async[T any](fn func(func(T))) Future[T] {
var result T
var done atomic.Bool
var resultReady atomic.Bool
var callbacks []func(T)
var mutex sync.Mutex
return func(chain func(T)) {
mutex.Lock()
if resultReady.Load() {
mutex.Unlock()
chain(result)
return
}
callbacks = append(callbacks, chain)
if !done.Swap(true) {
mutex.Unlock()
loop := Exec().L
var a *libuv.Async
var cb libuv.AsyncCb
a, cb = cbind.BindF[libuv.Async, libuv.AsyncCb](func(a *libuv.Async) {
a.Close(nil)
mutex.Lock()
resultReady.Store(true)
currentCallbacks := callbacks
callbacks = nil
mutex.Unlock()
for _, callback := range currentCallbacks {
callback(result)
}
})
loop.Async(a, cb)
fn(func(v T) {
result = v
a.Send()
})
} else {
mutex.Unlock()
}
}
2024-09-04 17:08:08 +08:00
}
// -----------------------------------------------------------------------------
2024-09-05 16:00:26 +08:00
func Race[T1 any](futures ...Future[T1]) Future[T1] {
2024-09-04 17:08:08 +08:00
return Async(func(resolve func(T1)) {
2024-09-05 16:00:26 +08:00
done := atomic.Bool{}
for _, future := range futures {
2024-09-08 20:27:05 +08:00
future.Then(func(v T1) {
2024-09-05 16:00:26 +08:00
if !done.Swap(true) {
// Just resolve the first one.
2024-09-05 16:00:26 +08:00
resolve(v)
}
2024-09-04 17:08:08 +08:00
})
}
})
}
2024-09-05 16:00:26 +08:00
func All[T1 any](futures ...Future[T1]) Future[[]T1] {
2024-09-04 17:08:08 +08:00
return Async(func(resolve func([]T1)) {
2024-09-05 16:00:26 +08:00
n := len(futures)
2024-09-04 17:08:08 +08:00
results := make([]T1, n)
2024-09-05 16:00:26 +08:00
var done uint32
for i, future := range futures {
2024-09-04 17:08:08 +08:00
i := i
2024-09-08 20:27:05 +08:00
future.Then(func(v T1) {
2024-09-05 16:00:26 +08:00
results[i] = v
if atomic.AddUint32(&done, 1) == uint32(n) {
// All done.
2024-09-05 16:00:26 +08:00
resolve(results)
}
2024-09-04 17:08:08 +08:00
})
}
})
}