2020-06-06 14:29:44 +00:00
|
|
|
# License
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
libco is released under the ISC license.
|
2015-12-30 06:41:46 +00:00
|
|
|
|
2020-06-06 14:29:44 +00:00
|
|
|
# Foreword
|
2017-10-25 01:50:54 +00:00
|
|
|
libco is a cross-platform, permissively licensed implementation of
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
cooperative-multithreading; a feature that is sorely lacking from the ISO C/C++
|
2020-06-06 14:29:44 +00:00
|
|
|
standard.
|
|
|
|
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
The library is designed for maximum speed and portability, and not for safety or
|
|
|
|
features. If safety or extra functionality is desired, a wrapper API can easily
|
2020-06-06 14:29:44 +00:00
|
|
|
be written to encapsulate all library functions.
|
|
|
|
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
Behavior of executing operations that are listed as not permitted below result
|
|
|
|
in undefined behavior. They may work anyway, they may cause undesired / unknown
|
2020-06-06 14:29:44 +00:00
|
|
|
behavior, or they may crash the program entirely.
|
|
|
|
|
2015-12-30 06:41:46 +00:00
|
|
|
The goal of this library was to simplify the base API as much as possible,
|
|
|
|
implementing only that which cannot be implemented using pure C. Additional
|
|
|
|
functionality after this would only complicate ports of this library to new
|
|
|
|
platforms.
|
|
|
|
|
2020-06-06 14:29:44 +00:00
|
|
|
# Porting
|
2015-12-30 06:41:46 +00:00
|
|
|
This document is included as a reference for porting libco. Please submit any
|
|
|
|
ports you create to me, so that libco can become more useful. Please note that
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
since libco is permissively licensed, you must submit your code as a work of the
|
|
|
|
public domain in order for it to be included in the official distribution.
|
2020-06-06 14:29:44 +00:00
|
|
|
|
Update to v105r1 release.
byuu says:
Changelog:
- higan: readded support for soft-reset to Famicom, Super Famicom,
Mega Drive cores (work in progress)
- handhelds lack soft reset obviously
- the PC Engine also lacks a physical reset button
- the Master System's reset button acts like a gamepad button, so
can't show up in the menu
- Mega Drive: power cycle wasn't initializing CPU (M68K) or APU (Z80)
RAM
- Super Famicom: fix SPC700 opcode 0x3b regression; fixes Majuu Ou
[Jonas Quinn]
- Super Famicom: fix SharpRTC save regression; fixes Dai Kaijuu
Monogatari II's real-time clock [Talarubi]
- Super Famicom: fix EpsonRTC save regression; fixes Tengai Makyou
Zero's real-time clock [Talarubi]
- Super Famicom: removed `*::init()` functions, as they were never used
- Super Famicom: removed all but two `*::load()` functions, as they
were not used
- higan: added option to auto-save backup RAM every five seconds
(enabled by default)
- this is in case the emulator crashes, or there's a power outage;
turn it off under advanced settings if you want
- libco: updated license from public domain to ISC, for consistency
with nall, ruby, hiro
- nall: Linux compiler defaults to g++; override with g++-version if
g++ is <= 4.8
- FreeBSD compiler default is going to remain g++49 until my dev
box OS ships with g++ >= 4.9
Errata: I have weird RAM initialization constants, thanks to hex_usr
and onethirdxcubed for both finding this:
http://wiki.nesdev.com/w/index.php?title=CPU_power_up_state&diff=11711&oldid=11184
I'll remove this in the next WIP.
2017-11-06 22:05:54 +00:00
|
|
|
Full credit will be given in the source code of the official release. Please
|
|
|
|
do not bother submitting code to me under any other license -- including GPL,
|
|
|
|
LGPL, BSD or CC -- I am not interested in creating a library with multiple
|
|
|
|
different licenses depending on which targets are used.
|
2020-06-06 14:29:44 +00:00
|
|
|
|
|
|
|
Note that there are a variety of compile-time options in `settings.h`,
|
|
|
|
so if you want to use libco on a platform where it is not supported by default,
|
|
|
|
you may be able to configure the implementation appropriately without having
|
|
|
|
to make a whole new port.
|
|
|
|
|
|
|
|
# Synopsis
|
|
|
|
```c
|
|
|
|
typedef void* cothread_t;
|
|
|
|
|
|
|
|
cothread_t co_active();
|
|
|
|
cothread_t co_create(unsigned int heapsize, void (*coentry)(void));
|
|
|
|
void co_delete(cothread_t cothread);
|
|
|
|
void co_switch(cothread_t cothread);
|
|
|
|
```
|
|
|
|
|
|
|
|
# Usage
|
|
|
|
## cothread_t
|
|
|
|
```c
|
|
|
|
typedef void* cothread_t;
|
|
|
|
```
|
|
|
|
Handle to cothread.
|
|
|
|
|
|
|
|
Handle must be of type `void*`.
|
|
|
|
|
2021-01-08 13:04:01 +00:00
|
|
|
A value of null (0) indicates an uninitialized or invalid handle, whereas a
|
|
|
|
non-zero value indicates a valid handle. A valid handle is backed by execution
|
|
|
|
state to which the execution can be co_switch()ed to.
|
2020-06-06 14:29:44 +00:00
|
|
|
|
|
|
|
## co_active
|
|
|
|
```c
|
|
|
|
cothread_t co_active();
|
|
|
|
```
|
|
|
|
Return handle to current cothread.
|
|
|
|
|
2021-01-08 13:04:01 +00:00
|
|
|
Note that the handle is valid even if the function is called from a non-cothread
|
|
|
|
context. To achieve this, we save the execution state in an internal buffer,
|
|
|
|
instead of using the user-provided memory. Since this handle is valid, it can
|
|
|
|
be used to co_switch to this context from another cothread. In multi-threaded
|
|
|
|
applications, make sure to not switch non-cothread context across CPU cores,
|
|
|
|
to prevent any possible conflicts with the OS scheduler.
|
2020-06-06 14:29:44 +00:00
|
|
|
|
|
|
|
## co_derive
|
|
|
|
```c
|
|
|
|
cothread_t co_derive(void* memory,
|
|
|
|
unsigned int heapsize,
|
|
|
|
void (*coentry)(void));
|
|
|
|
```
|
|
|
|
Initializes new cothread.
|
|
|
|
|
|
|
|
This function is identical to `co_create`, only it attempts to use the provided
|
2019-02-27 12:02:30 +00:00
|
|
|
memory instead of allocating new memory on the heap. Please note that certain
|
|
|
|
implementations (currently only Windows Fibers) cannot be created using existing
|
|
|
|
memory, and as such, this function will fail.
|
|
|
|
|
2020-06-06 14:29:44 +00:00
|
|
|
## co_create
|
|
|
|
```c
|
|
|
|
cothread_t co_create(unsigned int heapsize,
|
|
|
|
void (*coentry)(void));
|
|
|
|
```
|
|
|
|
Create new cothread.
|
|
|
|
|
|
|
|
`heapsize` is the amount of memory allocated for the cothread stack, specified
|
2015-12-30 06:41:46 +00:00
|
|
|
in bytes. This is unfortunately impossible to make fully portable. It is
|
2020-06-06 14:29:44 +00:00
|
|
|
recommended to specify sizes using `n * sizeof(void*)`. It is better to err
|
2015-12-30 06:41:46 +00:00
|
|
|
on the side of caution and allocate more memory than will be needed to ensure
|
|
|
|
compatibility with other platforms, within reason. A typical heapsize for a
|
2020-06-06 14:29:44 +00:00
|
|
|
32-bit architecture is ~1MB.
|
|
|
|
|
2015-12-30 06:41:46 +00:00
|
|
|
When the new cothread is first called, program execution jumps to coentry.
|
|
|
|
This function does not take any arguments, due to portability issues with
|
|
|
|
passing function arguments. However, arguments can be simulated by the use
|
2020-06-06 14:29:44 +00:00
|
|
|
of global variables, which can be set before the first call to each cothread.
|
|
|
|
|
|
|
|
`coentry()` must not return, and should end with an appropriate `co_switch()`
|
|
|
|
statement. Behavior is undefined if entry point returns normally.
|
|
|
|
|
2015-12-30 06:41:46 +00:00
|
|
|
Library is responsible for allocating cothread stack memory, to free
|
|
|
|
the user from needing to allocate special memory capable of being used
|
2020-06-06 14:29:44 +00:00
|
|
|
as program stack memory on platforms where this is required.
|
|
|
|
|
|
|
|
User is always responsible for deleting cothreads with `co_delete()`.
|
|
|
|
|
|
|
|
Return value of `null` (0) indicates cothread creation failed.
|
|
|
|
|
|
|
|
## co_delete
|
|
|
|
```c
|
|
|
|
void co_delete(cothread_t cothread);
|
|
|
|
```
|
|
|
|
Delete specified cothread.
|
|
|
|
|
|
|
|
`null` (0) or invalid cothread handle is not allowed.
|
2015-12-30 06:41:46 +00:00
|
|
|
|
|
|
|
Passing handle of active cothread to this function is not allowed.
|
|
|
|
|
2020-06-06 14:29:44 +00:00
|
|
|
Passing handle of primary cothread is not allowed.
|
|
|
|
|
2021-01-11 11:22:31 +00:00
|
|
|
## co_serializable
|
|
|
|
|
|
|
|
```c
|
|
|
|
int co_serializable(void);
|
|
|
|
```
|
|
|
|
|
|
|
|
Returns non-zero if the implementation keeps the entire coroutine state in the
|
|
|
|
buffer passed to `co_derive()`. That is, if `co_serializable()` returns
|
|
|
|
non-zero, and if your cothread does not modify the heap or any process-wide
|
|
|
|
state, then you can "snapshot" the cothread's state by taking a copy of the
|
|
|
|
buffer originally passed to `co_derive()`, and "restore" a previous state
|
|
|
|
by copying the snapshot back into the buffer it came from.
|
|
|
|
|
2020-06-06 14:29:44 +00:00
|
|
|
## co_switch
|
|
|
|
```c
|
|
|
|
void co_switch(cothread_t cothread);
|
|
|
|
```
|
|
|
|
Switch to specified cothread.
|
|
|
|
|
|
|
|
`null` (0) or invalid cothread handle is not allowed.
|
|
|
|
|
|
|
|
Passing handle of active cothread to this function is not allowed.
|