1 #include <sys/types.h>
2 #include <sys/socket.h>
3 #include <sys/time.h>
4 #include <linux/qrtr.h>
5 #include <unistd.h>
6 #include <stdlib.h>
7 #include <string.h>
8 #include <stdio.h>
9 #include <errno.h>
10 #include <poll.h>
11
12 #include "libqrtr.h"
13 #include "logging.h"
14 #include "ns.h"
15
qrtr_getname(int sock,struct sockaddr_qrtr * sq)16 static int qrtr_getname(int sock, struct sockaddr_qrtr *sq)
17 {
18 socklen_t sl = sizeof(*sq);
19 int rc;
20
21 rc = getsockname(sock, (void *)sq, &sl);
22 if (rc) {
23 PLOGE("getsockname()");
24 return -1;
25 }
26
27 if (sq->sq_family != AF_QIPCRTR || sl != sizeof(*sq))
28 return -1;
29
30 return 0;
31 }
32
qrtr_open(int rport)33 int qrtr_open(int rport)
34 {
35 struct timeval tv;
36 int sock;
37 int rc;
38
39 sock = socket(AF_QIPCRTR, SOCK_DGRAM, 0);
40 if (sock < 0) {
41 PLOGE("socket(AF_QIPCRTR)");
42 return -1;
43 }
44
45 tv.tv_sec = 1;
46 tv.tv_usec = 0;
47
48 rc = setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv));
49 if (rc) {
50 PLOGE("setsockopt(SO_RCVTIMEO)");
51 goto err;
52 }
53
54 if (rport != 0) {
55 struct sockaddr_qrtr sq = {};
56
57 sq.sq_family = AF_QIPCRTR;
58 sq.sq_node = 1;
59 sq.sq_port = rport;
60
61 rc = bind(sock, (void *)&sq, sizeof(sq));
62 if (rc < 0) {
63 PLOGE("bind(%d)", rport);
64 goto err;
65 }
66 }
67
68 return sock;
69 err:
70 close(sock);
71 return -1;
72 }
73
qrtr_close(int sock)74 void qrtr_close(int sock)
75 {
76 close(sock);
77 }
78
qrtr_sendto(int sock,uint32_t node,uint32_t port,const void * data,unsigned int sz)79 int qrtr_sendto(int sock, uint32_t node, uint32_t port, const void *data, unsigned int sz)
80 {
81 struct sockaddr_qrtr sq = {};
82 int rc;
83
84 sq.sq_family = AF_QIPCRTR;
85 sq.sq_node = node;
86 sq.sq_port = port;
87
88 rc = sendto(sock, data, sz, 0, (void *)&sq, sizeof(sq));
89 if (rc < 0) {
90 PLOGE("sendto()");
91 return -1;
92 }
93
94 return 0;
95 }
96
qrtr_new_server(int sock,uint32_t service,uint16_t version,uint16_t instance)97 int qrtr_new_server(int sock, uint32_t service, uint16_t version, uint16_t instance)
98 {
99 struct qrtr_ctrl_pkt pkt;
100 struct sockaddr_qrtr sq;
101
102 if (qrtr_getname(sock, &sq))
103 return -1;
104
105 memset(&pkt, 0, sizeof(pkt));
106
107 pkt.cmd = cpu_to_le32(QRTR_TYPE_NEW_SERVER);
108 pkt.server.service = cpu_to_le32(service);
109 pkt.server.instance = cpu_to_le32(instance << 8 | version);
110
111 return qrtr_sendto(sock, sq.sq_node, QRTR_PORT_CTRL, &pkt, sizeof(pkt));
112 }
113
qrtr_remove_server(int sock,uint32_t service,uint16_t version,uint16_t instance)114 int qrtr_remove_server(int sock, uint32_t service, uint16_t version, uint16_t instance)
115 {
116 struct qrtr_ctrl_pkt pkt;
117 struct sockaddr_qrtr sq;
118
119 if (qrtr_getname(sock, &sq))
120 return -1;
121
122 memset(&pkt, 0, sizeof(pkt));
123
124 pkt.cmd = cpu_to_le32(QRTR_TYPE_DEL_SERVER);
125 pkt.server.service = cpu_to_le32(service);
126 pkt.server.instance = cpu_to_le32(instance << 8 | version);
127 pkt.server.node = cpu_to_le32(sq.sq_node);
128 pkt.server.port = cpu_to_le32(sq.sq_port);
129
130 return qrtr_sendto(sock, sq.sq_node, QRTR_PORT_CTRL, &pkt, sizeof(pkt));
131 }
132
qrtr_publish(int sock,uint32_t service,uint16_t version,uint16_t instance)133 int qrtr_publish(int sock, uint32_t service, uint16_t version, uint16_t instance)
134 {
135 return qrtr_new_server(sock, service, version, instance);
136 }
137
qrtr_bye(int sock,uint32_t service,uint16_t version,uint16_t instance)138 int qrtr_bye(int sock, uint32_t service, uint16_t version, uint16_t instance)
139 {
140 return qrtr_remove_server(sock, service, version, instance);
141 }
142
qrtr_new_lookup(int sock,uint32_t service,uint16_t version,uint16_t instance)143 int qrtr_new_lookup(int sock, uint32_t service, uint16_t version, uint16_t instance)
144 {
145 struct qrtr_ctrl_pkt pkt;
146 struct sockaddr_qrtr sq;
147
148 if (qrtr_getname(sock, &sq))
149 return -1;
150
151 memset(&pkt, 0, sizeof(pkt));
152
153 pkt.cmd = cpu_to_le32(QRTR_TYPE_NEW_LOOKUP);
154 pkt.server.service = cpu_to_le32(service);
155 pkt.server.instance = cpu_to_le32(instance << 8 | version);
156
157 return qrtr_sendto(sock, sq.sq_node, QRTR_PORT_CTRL, &pkt, sizeof(pkt));
158 }
159
qrtr_remove_lookup(int sock,uint32_t service,uint16_t version,uint16_t instance)160 int qrtr_remove_lookup(int sock, uint32_t service, uint16_t version, uint16_t instance)
161 {
162 struct qrtr_ctrl_pkt pkt;
163 struct sockaddr_qrtr sq;
164
165 if (qrtr_getname(sock, &sq))
166 return -1;
167
168 memset(&pkt, 0, sizeof(pkt));
169
170 pkt.cmd = cpu_to_le32(QRTR_TYPE_DEL_LOOKUP);
171 pkt.server.service = cpu_to_le32(service);
172 pkt.server.instance = cpu_to_le32(instance << 8 | version);
173 pkt.server.node = cpu_to_le32(sq.sq_node);
174 pkt.server.port = cpu_to_le32(sq.sq_port);
175
176 return qrtr_sendto(sock, sq.sq_node, QRTR_PORT_CTRL, &pkt, sizeof(pkt));
177 }
178
qrtr_poll(int sock,unsigned int ms)179 int qrtr_poll(int sock, unsigned int ms)
180 {
181 struct pollfd fds;
182
183 fds.fd = sock;
184 fds.revents = 0;
185 fds.events = POLLIN | POLLERR;
186
187 return poll(&fds, 1, ms);
188 }
189
qrtr_recv(int sock,void * buf,unsigned int bsz)190 int qrtr_recv(int sock, void *buf, unsigned int bsz)
191 {
192 int rc;
193
194 rc = recv(sock, buf, bsz, 0);
195 if (rc < 0)
196 PLOGE("recv()");
197 return rc;
198 }
199
qrtr_recvfrom(int sock,void * buf,unsigned int bsz,uint32_t * node,uint32_t * port)200 int qrtr_recvfrom(int sock, void *buf, unsigned int bsz, uint32_t *node, uint32_t *port)
201 {
202 struct sockaddr_qrtr sq;
203 socklen_t sl;
204 int rc;
205
206 sl = sizeof(sq);
207 rc = recvfrom(sock, buf, bsz, 0, (void *)&sq, &sl);
208 if (rc < 0) {
209 PLOGE("recvfrom()");
210 return rc;
211 }
212 if (node)
213 *node = sq.sq_node;
214 if (port)
215 *port = sq.sq_port;
216 return rc;
217 }
218
qrtr_decode(struct qrtr_packet * dest,void * buf,size_t len,const struct sockaddr_qrtr * sq)219 int qrtr_decode(struct qrtr_packet *dest, void *buf, size_t len,
220 const struct sockaddr_qrtr *sq)
221 {
222 const struct qrtr_ctrl_pkt *ctrl = buf;
223
224 if (sq->sq_port == QRTR_PORT_CTRL){
225 if (len < sizeof(*ctrl))
226 return -EMSGSIZE;
227
228 dest->type = le32_to_cpu(ctrl->cmd);
229 switch (dest->type) {
230 case QRTR_TYPE_BYE:
231 dest->node = le32_to_cpu(ctrl->client.node);
232 break;
233 case QRTR_TYPE_DEL_CLIENT:
234 dest->node = le32_to_cpu(ctrl->client.node);
235 dest->port = le32_to_cpu(ctrl->client.port);
236 break;
237 case QRTR_TYPE_NEW_SERVER:
238 case QRTR_TYPE_DEL_SERVER:
239 dest->node = le32_to_cpu(ctrl->server.node);
240 dest->port = le32_to_cpu(ctrl->server.port);
241 dest->service = le32_to_cpu(ctrl->server.service);
242 dest->version = le32_to_cpu(ctrl->server.instance) & 0xff;
243 dest->instance = le32_to_cpu(ctrl->server.instance) >> 8;
244 break;
245 default:
246 dest->type = 0;
247 }
248 } else {
249 dest->type = QRTR_TYPE_DATA;
250 dest->node = sq->sq_node;
251 dest->port = sq->sq_port;
252
253 dest->data = buf;
254 dest->data_len = len;
255 }
256
257 return 0;
258 }
259