v/vlib/coroutines/coroutines.c.v

88 lines
2.4 KiB
V

// Copyright (c) 2019-2024 Alexander Medvednikov. All rights reserved.
// Use of this source code is governed by an MIT license
// that can be found in the LICENSE file.
module coroutines
import v.util
import time
#flag -I @VEXEROOT/thirdparty/photon
#flag @VEXEROOT/thirdparty/photon/photonwrapper.so
#include "photonwrapper.h"
fn C.set_photon_thread_stack_allocator(fn (voidptr, int) voidptr, fn (voidptr, voidptr, int))
fn C.default_photon_thread_stack_alloc(voidptr, int) voidptr
fn C.default_photon_thread_stack_dealloc(voidptr, voidptr, int)
fn C.new_photon_work_pool(int) voidptr
fn C.delete_photon_work_pool()
fn C.init_photon_work_pool(int)
fn C.init_photon_manual(int, fn ())
fn C.init_photon_manual2(fn (), fn ())
fn C.photon_thread_create_and_migrate_to_work_pool(f voidptr, arg voidptr)
fn C.photon_thread_create(f voidptr, arg voidptr)
fn C.photon_thread_migrate()
// fn C.photon_thread_migrate(work_pool voidptr)
fn C.photon_init_default() int
fn C.photon_sleep_s(n int)
fn C.photon_sleep_ms(n int)
// sleep is coroutine-safe version of time.sleep()
pub fn sleep(duration time.Duration) {
C.photon_sleep_ms(duration.milliseconds())
}
fn alloc(_ voidptr, stack_size int) voidptr {
// println('## alloc called')
unsafe {
// thread_id := C.pthread_self()
// println('## Thread ID: $thread_id')
// $if gcboehm ? {
// mut sb := C.GC_stack_base{}
// C.GC_get_stack_base(&sb)
// C.GC_register_my_thread(&sb)
// // res = C.GC_register_my_thread(&sb)
// // println('## RES: $res')
// }
// NOTE: when using malloc (GC_MALLOC) we get a segfault
// when migrating from a new thread to a work pool thread
// stack_ptr := malloc(stack_size)
// stack_ptr := C.malloc(stack_size)
stack_ptr := C.default_photon_thread_stack_alloc(nil, stack_size)
$if gcboehm ? {
C.GC_add_roots(stack_ptr, charptr(stack_ptr) + stack_size)
}
return stack_ptr
}
}
fn dealloc(_ voidptr, stack_ptr voidptr, stack_size int) {
// println('## dealloc called')
unsafe {
$if gcboehm ? {
// C.GC_unregister_my_thread()
C.GC_remove_roots(stack_ptr, charptr(stack_ptr) + stack_size)
}
// free(stack_ptr)
C.default_photon_thread_stack_dealloc(nil, stack_ptr, stack_size)
}
}
fn init() {
C.set_photon_thread_stack_allocator(alloc, dealloc)
ret := C.photon_init_default()
if util.nr_jobs >= 1 {
C.init_photon_work_pool(util.nr_jobs)
}
if ret < 0 {
panic('failed to initialize coroutines via photon (ret=${ret})')
}
}