forked from sustrik/libmill
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathchan.h
113 lines (96 loc) · 4.01 KB
/
chan.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/*
Copyright (c) 2015 Martin Sustrik
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom
the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
IN THE SOFTWARE.
*/
#ifndef MILL_CHAN_INCLUDED
#define MILL_CHAN_INCLUDED
#include <stddef.h>
#include "debug.h"
#include "list.h"
#include "slist.h"
/* One of these structures is preallocated for every coroutine. */
struct mill_choosedata {
/* List of clauses in the 'choose' statement. */
struct mill_slist clauses;
/* 1 if there is 'otherwise' clause. 0 if there is not. */
int othws;
/* Deadline specified in 'deadline' clause. -1 if none. */
int64_t ddline;
/* Number of clauses that are immediately available. */
int available;
};
/* Channel endpoint. */
struct mill_ep {
/* Thanks to this flag we can cast from ep pointer to chan pointer. */
enum {MILL_SENDER, MILL_RECEIVER} type;
/* Sequence number of the choose operation being initialised. */
int seqnum;
/* Number of clauses referring to this endpoint within the choose
operation being initialised. */
int refs;
/* Number of refs already processed. */
int tmp;
/* List of clauses waiting for this endpoint. */
struct mill_list clauses;
};
/* Channel. */
struct mill_chan {
/* The size of the elements stored in the channel, in bytes. */
size_t sz;
/* Channel holds two lists, the list of clauses waiting to send and list
of clauses waiting to receive. */
struct mill_ep sender;
struct mill_ep receiver;
/* Number of open handles to this channel. */
int refcount;
/* 1 is chdone() was already called. 0 otherwise. */
int done;
/* The message buffer directly follows the chan structure. 'bufsz' specifies
the maximum capacity of the buffer. 'items' is the number of messages
currently in the buffer. 'first' is the index of the next message to
be received from the buffer. There's one extra element at the end of
the buffer used to store the message supplied by chdone() function. */
size_t bufsz;
size_t items;
size_t first;
/* Debugging info. */
struct mill_debug_chan debug;
};
/* This structure represents a single clause in a choose statement.
Similarly, both chs() and chr() each create a single clause. */
struct mill_clause {
/* Member of list of clauses waiting for a channel endpoint. */
struct mill_list_item epitem;
/* Linked list of clauses in the choose statement. */
struct mill_slist_item chitem;
/* The coroutine which created the clause. */
struct mill_cr *cr;
/* Channel endpoint the clause is waiting for. */
struct mill_ep *ep;
/* For out clauses, pointer to the value to send. NULL for in clauses. */
void *val;
/* The index to jump to when the clause is executed. */
int idx;
/* If 0, there's no peer waiting for the clause at the moment.
If 1, there is one. */
int available;
/* If 1, the clause is in the list of channel's senders/receivers. */
int used;
};
/* Returns pointer to the channel that contains specified endpoint. */
struct mill_chan *mill_getchan(struct mill_ep *ep);
#endif