-
Notifications
You must be signed in to change notification settings - Fork 89
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #370 from tock/libtock-folders
libtock: Rewrite and standardize API
- Loading branch information
Showing
613 changed files
with
13,275 additions
and
13,437 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,353 @@ | ||
Writing a libtock-c System Call Driver | ||
====================================== | ||
|
||
This guide covers how to implement support in `libtock` for a new system call | ||
driver. | ||
|
||
|
||
## Setup | ||
|
||
|
||
- `[name]`: String which identifies the system call driver. | ||
- `[category]`: The general type of the system call. | ||
|
||
The categories are: | ||
|
||
- `crypto`: Cryptography interfaces. | ||
- `display`: Screens and other displays. | ||
- `interface`: Human-computer interfaces. | ||
- `kernel`: Tock-specific kernel-level drivers. | ||
- `net`: Networking drivers. | ||
- `peripherals`: Interfaces for chip peripherals. | ||
- `sensors`: Sensors. | ||
- `storage`: Various storage drivers. | ||
|
||
|
||
## Syscall APIs | ||
|
||
All supported system calls (i.e., command, allow, subscribe, etc.) for a | ||
particular driver must be wrapped in simple functions. | ||
|
||
| Characteristic | Value | | ||
|------------------|-------------------------------------------------| | ||
| Location | `libtock/[category]/syscalls` | | ||
| Source File Name | `libtock/[category]/syscalls/[name]_syscalls.c` | | ||
| Header File Name | `libtock/[category]/syscalls/[name]_syscalls.h` | | ||
|
||
### Header File | ||
|
||
The `[name]_syscalls.h` file must contain a `#define` called `DRIVER_NUM_[NAME]` | ||
with the driver number. | ||
|
||
_All_ header files must be wrapped in `extern "C" { ... }` if the header file | ||
is used in a C++ app. | ||
|
||
#### Example: | ||
|
||
```c | ||
#pragma once | ||
|
||
#include "../../tock.h" | ||
|
||
#ifdef __cplusplus | ||
extern "C" { | ||
#endif | ||
|
||
#define DRIVER_NUM_[NAME] 0x00000 | ||
|
||
// Other defines and signatures go here. | ||
|
||
#ifdef __cplusplus | ||
} | ||
#endif | ||
``` | ||
|
||
### Upcalls | ||
|
||
Each supported upcall must have its own function. | ||
|
||
The signature is: | ||
|
||
```c | ||
returncode_t libtock_[name]_set_upcall(subscribe_upcall callback, void* opaque); | ||
``` | ||
|
||
If only one upcall is supported, the function name must be `[name]_set_upcall`. | ||
|
||
If more than one upcall is supported, the function names must start with | ||
`libtock_[name]_set_upcall_` followed by a description of what the upcall is | ||
used for. | ||
|
||
|
||
#### Example: | ||
|
||
```c | ||
returncode_t libtock_[name]_set_upcall(subscribe_upcall callback, void* opaque) { | ||
subscribe_return_t sval = subscribe(DRIVER_NUM_[NAME], 0, callback, opaque); | ||
return tock_subscribe_return_to_returncode(sval); | ||
} | ||
``` | ||
|
||
### Allows | ||
|
||
Each supported allow (readonly, readwrite, userspace_read) must have its own | ||
function. | ||
|
||
The signature is: | ||
|
||
- Read-Only Allow: | ||
```c | ||
returncode_t libtock_[name]_set_readonly_allow(const uint8_t* buffer, uint32_t len); | ||
``` | ||
|
||
- Read-Write Allow: | ||
```c | ||
returncode_t libtock_[name]_set_readwrite_allow(uint8_t* buffer, uint32_t len); | ||
``` | ||
|
||
- Userspace Read Allow: | ||
```c | ||
returncode_t libtock_[name]_set_userspace_read_allow(uint8_t* buffer, uint32_t len); | ||
``` | ||
|
||
If only one allow is supported, the function name must be | ||
`libtock_[name]_set_[type]_allow`. | ||
|
||
If more than one allow is supported, the function names must start with | ||
`libtock_[name]_set_[type]_allow_` followed by a description of what the allow | ||
is used for. | ||
|
||
#### Example: | ||
|
||
```c | ||
returncode_t libtock_[name]_set_readonly_allow_[desc](const uint8_t* buffer, uint32_t len) { | ||
allow_ro_return_t aval = allow_readonly(DRIVER_NUM_[NAME], 0, (void*) buffer, len); | ||
return tock_allow_ro_return_to_returncode(aval); | ||
} | ||
``` | ||
|
||
```c | ||
returncode_t libtock_[name]_set_readwrite_allow_[desc](uint8_t* buffer, uint32_t len) { | ||
allow_rw_return_t aval = allow_readwrite(DRIVER_NUM_[NAME], 0, (void*) buffer, len); | ||
return tock_allow_rw_return_to_returncode(aval); | ||
} | ||
``` | ||
|
||
```c | ||
returncode_t libtock_[name]_set_userspace_read_allow_[desc](uint8_t* buffer, uint32_t len) { | ||
allow_userspace_r_return_t aval = allow_userspace_read(DRIVER_NUM_[NAME], 0, (void*) buffer, len); | ||
return tock_allow_userspace_r_return_to_returncode(aval); | ||
} | ||
``` | ||
|
||
### Commands | ||
|
||
Each supported command must have its own function. | ||
|
||
The signature is: | ||
|
||
```c | ||
returncode_t libtock_[name]_command_[desc](<arguments>); | ||
``` | ||
|
||
For every command, the function names must start with `libtock_[name]_command_` | ||
followed by a description of what the command is used for. | ||
|
||
### Example: | ||
|
||
```c | ||
returncode_t libtock_[name]_command_[desc](void) { | ||
syscall_return_t cval = command(DRIVER_NUM_[NAME], 1, 0, 0); | ||
return tock_command_return_novalue_to_returncode(cval); | ||
} | ||
``` | ||
|
||
|
||
### Exists | ||
|
||
There must be a function to check for syscall driver existence. | ||
|
||
Signature: | ||
|
||
``` | ||
bool libtock_[name]_exists(void); | ||
``` | ||
|
||
Example: | ||
|
||
```c | ||
bool libtock_[name]_exists(void) { | ||
return driver_exists(DRIVER_NUM_[NAME]); | ||
} | ||
``` | ||
|
||
|
||
## Asynchronous APIs | ||
|
||
All common system call operations should have asynchronous versions. | ||
|
||
These asynchronous APIs must not use or include any internal/global state. | ||
|
||
| Characteristic | Value | | ||
|------------------|-------------------------------| | ||
| Location | `libtock/[category]` | | ||
| Source File Name | `libtock/[category]/[name].c` | | ||
| Header File Name | `libtock/[category]/[name].h` | | ||
|
||
### Header Files | ||
|
||
The `[name].h` header file must look like: | ||
|
||
```c | ||
#pragma once | ||
|
||
#include "../tock.h" | ||
#include "syscalls/[name]_syscalls.h" | ||
|
||
#ifdef __cplusplus | ||
extern "C" { | ||
#endif | ||
|
||
// Function signatures go here. | ||
|
||
#ifdef __cplusplus | ||
} | ||
#endif | ||
``` | ||
|
||
The `[name].h` header file must include the syscalls header. | ||
|
||
### Defining a Callback for Asynchronous Operations | ||
|
||
For every type of callback the system call can generate, the library must have a | ||
function signature defined as a type. The format is: | ||
|
||
```c | ||
typedef void (*libtock_[name]_callback_[desc])(returncode_t, int); | ||
``` | ||
### Functions for Operations | ||
Every operation the libtock library should expose must be defined. The return | ||
type should generally be `returncode_t`. As these are asynchronous operations, | ||
they should have the last argument be a callback function pointer. | ||
```c | ||
returncode_t libtock_[name]_[desc](<arguments>, libtock_[name]_callback_[desc] cb); | ||
``` | ||
|
||
### Example | ||
|
||
|
||
For example, a library called "sensor" with a sensor read operation should look | ||
like this: | ||
|
||
Define a suitable callback for the library: | ||
|
||
```c | ||
typedef void (*libtock_sensor_callback_reading)(returncode_t, int); | ||
``` | ||
Define a upcall function to be passed to the kernel: | ||
```c | ||
static void sensor_temp_upcall(int ret, | ||
int val, | ||
__attribute__ ((unused)) int unused0, | ||
void* opaque) { | ||
libtock_sensor_callback_reading cb = (libtock_sensor_callback_reading) opaque; | ||
cb(ret, val); | ||
} | ||
``` | ||
|
||
Define an external function for applications to use to run the asynchronous | ||
operation: | ||
|
||
```c | ||
returncode_t libtock_sensor_read(libtock_sensor_callback_reading cb) { | ||
returncode_t ret; | ||
|
||
ret = libtock_sensor_set_upcall(sensor_temp_upcall, cb); | ||
if (ret != RETURNCODE_SUCCESS) return ret; | ||
|
||
ret = libtock_sensor_command_read(); | ||
return ret; | ||
} | ||
``` | ||
## Synchronous APIs | ||
Most system call interfaces will want to provide a synchronous API as well. | ||
| Characteristic | Value | | ||
|------------------|------------------------------------| | ||
| Location | `libtock-sync/[category]` | | ||
| Source File Name | `libtock-sync/[category]/[name].c` | | ||
| Header File Name | `libtock-sync/[category]/[name].h` | | ||
### Header Files | ||
The libtock-sync `[name].h` header file must look like: | ||
```c | ||
#pragma once | ||
#include <libtock/tock.h> | ||
#include <libtock/[category]/[name].h> | ||
#ifdef __cplusplus | ||
extern "C" { | ||
#endif | ||
// Function signatures go here. | ||
#ifdef __cplusplus | ||
} | ||
#endif | ||
``` | ||
|
||
### Synchronous APIs | ||
|
||
For our sensor example: | ||
|
||
Create a struct to be set by the asynchronous callback: | ||
|
||
```c | ||
struct data { | ||
bool fired; | ||
int val; | ||
returncode_t result; | ||
}; | ||
|
||
static struct data result = { .fired = false }; | ||
``` | ||
Define a static callback function to pass to the asynchronous implementation: | ||
```c | ||
static void sensor_cb(returncode_t ret, int val) { | ||
result.val = val; | ||
result.fired = true; | ||
result.result = ret; | ||
} | ||
``` | ||
|
||
Define the external function in the `libtocksync_` name space for the sync | ||
operation: | ||
|
||
```c | ||
returncode_t libtocksync_sensor_read(int* val) { | ||
int err; | ||
result.fired = false; | ||
|
||
err = libtock_sensor_read(sensor_cb); | ||
if (err != RETURNCODE_SUCCESS) return err; | ||
|
||
// Wait for the callback. | ||
yield_for(&result.fired); | ||
if (result.result != RETURNCODE_SUCCESS) return result.result; | ||
|
||
*val = result.val; | ||
return RETURNCODE_SUCCESS; | ||
} | ||
``` |
Oops, something went wrong.