chiark / gitweb /
udp, polypath: Log `experiencing success receiving', etc.
[secnet.git] / comm-common.h
1
2 #ifndef COMM_COMMON_H
3 #define COMM_COMMON_H
4
5 #include "secnet.h"
6
7 /*----- for all comms -----*/
8
9 struct comm_notify_entry {
10     comm_notify_fn *fn;
11     void *state;
12     LIST_ENTRY(comm_notify_entry) entry;
13 };
14 LIST_HEAD(comm_notify_list, comm_notify_entry) notify;
15
16 struct commcommon { /* must be first so that void* is comm_common* */
17     closure_t cl;
18     struct comm_if ops;
19     struct cloc loc;
20     struct comm_notify_list notify;
21     struct buffer_if *rbuf;
22 };
23
24 void comm_request_notify(void *commst, void *nst, comm_notify_fn *fn);
25 void comm_release_notify(void *commst, void *nst, comm_notify_fn *fn);
26
27 bool_t comm_notify(struct comm_notify_list *notify, struct buffer_if *buf,
28                    const struct comm_addr *ca);
29   /* Either: returns True, with message delivered and buffer freed.
30    * Or: False, if no-one wanted it - buffer still allocd'd.
31    * Ie, like comm_notify_fn. */
32
33 void comm_apply(struct commcommon *cc);
34
35 #define COMM_APPLY(st,cc,prefix,desc,loc)               \
36     (st)=safe_malloc(sizeof(*(st)), desc "_apply");     \
37     (cc)->loc=loc;                                      \
38     (cc)->cl.description=desc;                          \
39     (cc)->ops.sendmsg=prefix##sendmsg;                  \
40     (cc)->ops.addr_to_string=prefix##addr_to_string;    \
41     comm_apply((cc))
42    /* Expects in scope: prefix##sendmsg, prefix##addr_to_string. */
43
44 #define COMM_APPLY_STANDARD(st,cc,desc,args)                            \
45     item_t *item=list_elem(args,0);                                     \
46     if (!item || item->type!=t_dict) {                                  \
47         cfgfatal((cc)->loc,desc,"first argument must be a dictionary\n"); \
48     }                                                                   \
49     dict_t *d=item->data.dict;                                          \
50     (cc)->rbuf=find_cl_if(d,"buffer",CL_BUFFER,True,desc,(cc)->loc)
51
52 /*----- for udp-based comms -----*/
53
54 #define UDP_MAX_SOCKETS 3 /* 2 ought to do really */
55
56 #define MAX_AF (AF_INET6>AF_INET?AF_INET6:AF_INET)
57
58 struct udpsock {
59     union iaddr addr;
60     int fd;
61     bool_t experienced[/*0=recv,1=send*/2][MAX_AF+1][/*success?*/2];
62 };
63
64 struct udpsocks {
65     int n_socks;
66     struct udpsock socks[UDP_MAX_SOCKETS];
67     /* private for udp_socks_* */
68     struct udpcommon *uc; /* link to parent, for cfg, notify list, etc. */
69     struct poll_interest *interest;
70     const char *desc;
71 };
72
73 struct udpcommon {
74     struct commcommon cc;
75     int port;
76     string_t authbind;
77     bool_t use_proxy;
78     union iaddr proxy;
79 };
80
81 bool_t udp_make_socket(struct udpcommon *uc, struct udpsock *us,
82                        int failmsgclass);
83   /* Caller should have filled in ->addr.  Fills in us->fd,
84      ->experienced; updates ->addr.  Logs any errors with lg_[v]perror. */
85 bool_t udp_import_socket(struct udpcommon *uc, struct udpsock *us,
86                          int failmsgclass, int fd);
87   /* Like udp_make_socket, but caller provides fd.  fd is not closed
88      on error */
89
90 void udp_destroy_socket(struct udpcommon *uc, struct udpsock *us);
91   /* Idempotent.  No errors are possible. */
92
93 const char *af_name(int af);
94 void udp_sock_experienced(struct log_if *lg, struct udpcommon *uc,
95                           struct udpsocks *socks, struct udpsock *us,
96                           bool_t recvsend, int af /* 0 means any */,
97                           int r, int errnoval);
98
99 void udp_socks_register(struct udpcommon *uc, struct udpsocks *socks,
100                         const char *desc);
101 void udp_socks_deregister(struct udpcommon *uc, struct udpsocks *socks);
102 void udp_socks_childpersist(struct udpcommon *uc, struct udpsocks *socks);
103
104 #define UDP_APPLY_STANDARD(st,uc,desc)                                  \
105     (uc)->use_proxy=False;                                              \
106     (uc)->authbind=NULL;                                                \
107     (uc)->port=dict_read_number(d,"port",False,"udp",(uc)->cc.loc,0)
108     /* Expects in scope:  dict_t *d=...; */
109
110 #endif /*COMM_COMMON_H*/