blob: 749e346a39a9b008519424a618278cfe5a735f78 [file] [log] [blame]
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -06001/******************************************************************************
2
3 L I B R M N E T C T L . C
4
Subash Abhinov Kasiviswanathancf715082015-04-14 13:51:59 -06005Copyright (c) 2013-2015, The Linux Foundation. All rights reserved.
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -06006
7Redistribution and use in source and binary forms, with or without
8modification, are permitted provided that the following conditions are
9met:
10 * Redistributions of source code must retain the above copyright
11 notice, this list of conditions and the following disclaimer.
12 * Redistributions in binary form must reproduce the above
13 copyright notice, this list of conditions and the following
14 disclaimer in the documentation and/or other materials provided
15 with the distribution.
16 * Neither the name of The Linux Foundation nor the names of its
17 contributors may be used to endorse or promote products derived
18 from this software without specific prior written permission.
19
20THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
21WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
22MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
23ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
24BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
27BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
28WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
29OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
30IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31
32******************************************************************************/
33
34/*!
35* @file librmnetctl.c
Subash Abhinov Kasiviswanathancf715082015-04-14 13:51:59 -060036* @brief rmnet control API's implementation file
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -060037*/
38
39/*===========================================================================
40 INCLUDE FILES
41===========================================================================*/
42
43#include <sys/socket.h>
44#include <stdint.h>
45#include <linux/netlink.h>
46#include <string.h>
47#include <stdio.h>
48#include <unistd.h>
49#include <stdlib.h>
50#include <linux/rmnet_data.h>
51#include "librmnetctl_hndl.h"
52#include "librmnetctl.h"
53
Harout Hedeshian04d69732013-10-02 09:29:11 -060054#ifdef USE_GLIB
55#include <glib.h>
56#define strlcpy g_strlcpy
57#endif
58
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -060059#define RMNETCTL_SOCK_FLAG 0
60#define ROOT_USER_ID 0
61#define MIN_VALID_PROCESS_ID 0
62#define MIN_VALID_SOCKET_FD 0
63#define KERNEL_PROCESS_ID 0
64#define UNICAST 0
65#define MAX_BUF_SIZE sizeof(struct nlmsghdr) + sizeof(struct rmnet_nl_msg_s)
Harout Hedeshian77825572014-02-17 10:47:14 +020066#define INGRESS_FLAGS_MASK (RMNET_INGRESS_FIX_ETHERNET | \
67 RMNET_INGRESS_FORMAT_MAP | \
68 RMNET_INGRESS_FORMAT_DEAGGREGATION | \
69 RMNET_INGRESS_FORMAT_DEMUXING | \
Sivan Reinsteinc22a0d82014-07-13 16:58:21 +030070 RMNET_INGRESS_FORMAT_MAP_COMMANDS | \
71 RMNET_INGRESS_FORMAT_MAP_CKSUMV3)
Harout Hedeshian77825572014-02-17 10:47:14 +020072#define EGRESS_FLAGS_MASK (RMNET_EGRESS_FORMAT__RESERVED__ | \
73 RMNET_EGRESS_FORMAT_MAP | \
74 RMNET_EGRESS_FORMAT_AGGREGATION | \
Sivan Reinsteina3c57172014-09-10 14:11:58 +030075 RMNET_EGRESS_FORMAT_MUXING | \
76 RMNET_EGRESS_FORMAT_MAP_CKSUMV3)
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -060077
Harout Hedeshian77825572014-02-17 10:47:14 +020078#define min(a, b) (((a) < (b)) ? (a) : (b))
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -060079/*===========================================================================
80 LOCAL FUNCTION DEFINITIONS
81===========================================================================*/
82/*!
83* @brief Synchronous method to send and receive messages to and from the kernel
84* using netlink sockets
85* @details Increments the transaction id for each message sent to the kernel.
86* Sends the netlink message to the kernel and receives the response from the
87* kernel.
88* @param *hndl RmNet handle for this transaction
89* @param request Message to be sent to the kernel
90* @param response Message received from the kernel
91* @return RMNETCTL_API_SUCCESS if successfully able to send and receive message
92* from the kernel
93* @return RMNETCTL_API_ERR_HNDL_INVALID if RmNet handle for the transaction was
94* NULL
95* @return RMNETCTL_API_ERR_REQUEST_NULL not enough memory to create buffer for
96* sending the message
97* @return RMNETCTL_API_ERR_MESSAGE_SEND if could not send the message to kernel
98* @return RMNETCTL_API_ERR_MESSAGE_RECEIVE if could not receive message from the
99* kernel
100* @return RMNETCTL_API_ERR_MESSAGE_TYPE if the request and response type do not
101* match
102*/
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700103static uint16_t rmnetctl_transact(rmnetctl_hndl_t *hndl,
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600104 struct rmnet_nl_msg_s *request,
105 struct rmnet_nl_msg_s *response) {
106 uint8_t *request_buf, *response_buf;
107 struct nlmsghdr *nlmsghdr_val;
108 struct rmnet_nl_msg_s *rmnet_nl_msg_s_val;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700109 ssize_t bytes_read = -1;
110 uint16_t return_code = RMNETCTL_API_ERR_HNDL_INVALID;
111 struct sockaddr_nl* __attribute__((__may_alias__)) saddr_ptr;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600112 request_buf = NULL;
113 response_buf = NULL;
114 nlmsghdr_val = NULL;
115 rmnet_nl_msg_s_val = NULL;
116 do {
117 if (!hndl){
118 break;
119 }
120 if (!request){
121 return_code = RMNETCTL_API_ERR_REQUEST_NULL;
122 break;
123 }
124 if (!response){
125 return_code = RMNETCTL_API_ERR_RESPONSE_NULL;
126 break;
127 }
128 request_buf = (uint8_t *)malloc(MAX_BUF_SIZE * sizeof(uint8_t));
129 if (!request_buf){
130 return_code = RMNETCTL_API_ERR_REQUEST_NULL;
131 break;
132 }
133
134 response_buf = (uint8_t *)malloc(MAX_BUF_SIZE * sizeof(uint8_t));
135 if (!response_buf) {
136 free(request_buf);
137 return_code = RMNETCTL_API_ERR_RESPONSE_NULL;
138 break;
139 }
140
141 nlmsghdr_val = (struct nlmsghdr *)request_buf;
142 rmnet_nl_msg_s_val = (struct rmnet_nl_msg_s *)NLMSG_DATA(request_buf);
143
144 memset(request_buf, 0, MAX_BUF_SIZE*sizeof(uint8_t));
145 memset(response_buf, 0, MAX_BUF_SIZE*sizeof(uint8_t));
146
147 nlmsghdr_val->nlmsg_seq = hndl->transaction_id;
148 nlmsghdr_val->nlmsg_pid = hndl->pid;
149 nlmsghdr_val->nlmsg_len = MAX_BUF_SIZE;
150
151 memcpy((void *)NLMSG_DATA(request_buf), request,
152 sizeof(struct rmnet_nl_msg_s));
153
154 rmnet_nl_msg_s_val->crd = RMNET_NETLINK_MSG_COMMAND;
155 hndl->transaction_id++;
156
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700157 saddr_ptr = &hndl->dest_addr;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600158 socklen_t addrlen = sizeof(struct sockaddr_nl);
159 if (sendto(hndl->netlink_fd,
160 request_buf,
161 MAX_BUF_SIZE,
162 RMNETCTL_SOCK_FLAG,
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700163 (struct sockaddr*)saddr_ptr,
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600164 sizeof(struct sockaddr_nl)) < 0) {
165 return_code = RMNETCTL_API_ERR_MESSAGE_SEND;
166 free(request_buf);
167 free(response_buf);
168 break;
169 }
170
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700171 saddr_ptr = &hndl->src_addr;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600172 bytes_read = recvfrom(hndl->netlink_fd,
173 response_buf,
174 MAX_BUF_SIZE,
175 RMNETCTL_SOCK_FLAG,
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700176 (struct sockaddr*)saddr_ptr,
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600177 &addrlen);
178 if (bytes_read < 0) {
179 return_code = RMNETCTL_API_ERR_MESSAGE_RECEIVE;
180 free(request_buf);
181 free(response_buf);
182 break;
183 }
184
185 memcpy(response, (void *)NLMSG_DATA(response_buf),
186 sizeof(struct rmnet_nl_msg_s));
187 if (sizeof(*response) < sizeof(struct rmnet_nl_msg_s)) {
188 return_code = RMNETCTL_API_ERR_RESPONSE_NULL;
189 free(request_buf);
190 free(response_buf);
191 break;
192 }
193
194 if (request->message_type != response->message_type) {
195 return_code = RMNETCTL_API_ERR_MESSAGE_TYPE;
196 free(request_buf);
197 free(response_buf);
198 break;
199 }
200 return_code = RMNETCTL_SUCCESS;
201 } while(0);
Harout Hedeshian77825572014-02-17 10:47:14 +0200202 free(request_buf);
203 free(response_buf);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600204 return return_code;
205}
206
207/*!
208* @brief Static function to check the dev name
209* @details Checks if the name is not NULL and if the name is less than the
210* RMNET_MAX_STR_LEN
211* @param dev_name Name of the device
212* @return RMNETCTL_SUCCESS if successful
213* @return RMNETCTL_INVALID_ARG if invalid arguments were passed to the API
214*/
215static inline int _rmnetctl_check_dev_name(const char *dev_name) {
216 int return_code = RMNETCTL_INVALID_ARG;
217 do {
218 if (!dev_name)
219 break;
220 if (strlen(dev_name) >= RMNET_MAX_STR_LEN)
221 break;
222 return_code = RMNETCTL_SUCCESS;
223 } while(0);
224 return return_code;
225}
226
227/*!
228* @brief Static function to check the string length after a copy
229* @details Checks if the string length is not lesser than zero and lesser than
230* RMNET_MAX_STR_LEN
231* @param str_len length of the string after a copy
232* @param error_code Status code of this operation
233* @return RMNETCTL_SUCCESS if successful
234* @return RMNETCTL_LIB_ERR if there was a library error. Check error_code
235*/
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700236static inline int _rmnetctl_check_len(size_t str_len, uint16_t *error_code) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600237 int return_code = RMNETCTL_LIB_ERR;
238 do {
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700239 if (str_len > RMNET_MAX_STR_LEN) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600240 *error_code = RMNETCTL_API_ERR_STRING_TRUNCATION;
241 break;
242 }
243 return_code = RMNETCTL_SUCCESS;
244 } while(0);
245 return return_code;
246}
247
248/*!
249* @brief Static function to check the response type
250* @details Checks if the response type of this message was return code
251* @param crd The crd field passed
252* @param error_code Status code of this operation
253* @return RMNETCTL_SUCCESS if successful
254* @return RMNETCTL_LIB_ERR if there was a library error. Check error_code
255*/
256static inline int _rmnetctl_check_code(int crd, uint16_t *error_code) {
257 int return_code = RMNETCTL_LIB_ERR;
258 do {
259 if (crd != RMNET_NETLINK_MSG_RETURNCODE) {
260 *error_code = RMNETCTL_API_ERR_RETURN_TYPE;
261 break;
262 }
263 return_code = RMNETCTL_SUCCESS;
264 } while(0);
265 return return_code;
266}
267
268/*!
269* @brief Static function to check the response type
270* @details Checks if the response type of this message was data
271* @param crd The crd field passed
272* @param error_code Status code of this operation
273* @return RMNETCTL_SUCCESS if successful
274* @return RMNETCTL_LIB_ERR if there was a library error. Check error_code
275*/
276static inline int _rmnetctl_check_data(int crd, uint16_t *error_code) {
277 int return_code = RMNETCTL_LIB_ERR;
278 do {
279 if (crd != RMNET_NETLINK_MSG_RETURNDATA) {
280 *error_code = RMNETCTL_API_ERR_RETURN_TYPE;
281 break;
282 }
283 return_code = RMNETCTL_SUCCESS;
284 } while(0);
285 return return_code;
286}
287
288/*!
289* @brief Static function to set the return value
290* @details Checks if the error_code from the transaction is zero for a return
291* code type message and sets the message type as RMNETCTL_SUCCESS
292* @param crd The crd field passed
293* @param error_code Status code of this operation
294* @return RMNETCTL_SUCCESS if successful
295* @return RMNETCTL_KERNEL_ERR if there was an error in the kernel.
296* Check error_code
297*/
298static inline int _rmnetctl_set_codes(int error_val, uint16_t *error_code) {
299 int return_code = RMNETCTL_KERNEL_ERR;
Harout Hedeshian77825572014-02-17 10:47:14 +0200300 if (error_val == RMNET_CONFIG_OK)
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600301 return_code = RMNETCTL_SUCCESS;
Harout Hedeshian77825572014-02-17 10:47:14 +0200302 else
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700303 *error_code = (uint16_t)error_val + RMNETCTL_KERNEL_FIRST_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600304 return return_code;
305}
306
307/*===========================================================================
308 EXPOSED API
309===========================================================================*/
310
311int rmnetctl_init(rmnetctl_hndl_t **hndl, uint16_t *error_code)
312{
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700313 pid_t pid = 0;
314 int netlink_fd = -1, return_code = RMNETCTL_LIB_ERR;
315 struct sockaddr_nl* __attribute__((__may_alias__)) saddr_ptr;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600316 do {
317 if ((!hndl) || (!error_code)){
318 return_code = RMNETCTL_INVALID_ARG;
319 break;
320 }
321
322 *hndl = (rmnetctl_hndl_t *)malloc(sizeof(rmnetctl_hndl_t));
323 if (!*hndl) {
324 *error_code = RMNETCTL_API_ERR_HNDL_INVALID;
325 break;
326 }
327
328 memset(*hndl, 0, sizeof(rmnetctl_hndl_t));
329
330 pid = getpid();
331 if (pid < MIN_VALID_PROCESS_ID) {
332 free(*hndl);
333 *error_code = RMNETCTL_INIT_ERR_PROCESS_ID;
334 break;
335 }
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700336 (*hndl)->pid = (uint32_t)pid;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600337 netlink_fd = socket(PF_NETLINK, SOCK_RAW, RMNET_NETLINK_PROTO);
338 if (netlink_fd < MIN_VALID_SOCKET_FD) {
339 free(*hndl);
340 *error_code = RMNETCTL_INIT_ERR_NETLINK_FD;
341 break;
342 }
343
344 (*hndl)->netlink_fd = netlink_fd;
345
346 memset(&(*hndl)->src_addr, 0, sizeof(struct sockaddr_nl));
347
348 (*hndl)->src_addr.nl_family = AF_NETLINK;
349 (*hndl)->src_addr.nl_pid = (*hndl)->pid;
350
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700351 saddr_ptr = &(*hndl)->src_addr;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600352 if (bind((*hndl)->netlink_fd,
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700353 (struct sockaddr*)saddr_ptr,
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600354 sizeof(struct sockaddr_nl)) < 0) {
355 close((*hndl)->netlink_fd);
356 free(*hndl);
357 *error_code = RMNETCTL_INIT_ERR_BIND;
358 break;
359 }
360
361 memset(&(*hndl)->dest_addr, 0, sizeof(struct sockaddr_nl));
362
363 (*hndl)->dest_addr.nl_family = AF_NETLINK;
364 (*hndl)->dest_addr.nl_pid = KERNEL_PROCESS_ID;
365 (*hndl)->dest_addr.nl_groups = UNICAST;
366
367 return_code = RMNETCTL_SUCCESS;
368 } while(0);
369 return return_code;
370}
371
372void rmnetctl_cleanup(rmnetctl_hndl_t *hndl)
373{
374 if (!hndl)
375 return;
376 close(hndl->netlink_fd);
377 free(hndl);
378}
379
380int rmnet_associate_network_device(rmnetctl_hndl_t *hndl,
381 const char *dev_name,
382 uint16_t *error_code,
383 uint8_t assoc_dev)
384{
385 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700386 size_t str_len = 0;
387 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600388 do {
389 if ((!hndl) || (!error_code) || _rmnetctl_check_dev_name(dev_name) ||
390 ((assoc_dev != RMNETCTL_DEVICE_ASSOCIATE) &&
391 (assoc_dev != RMNETCTL_DEVICE_UNASSOCIATE))) {
392 return_code = RMNETCTL_INVALID_ARG;
393 break;
394 }
395
396 if (assoc_dev == RMNETCTL_DEVICE_ASSOCIATE)
397 request.message_type = RMNET_NETLINK_ASSOCIATE_NETWORK_DEVICE;
398 else
399 request.message_type = RMNET_NETLINK_UNASSOCIATE_NETWORK_DEVICE;
400
401 request.arg_length = RMNET_MAX_STR_LEN;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700402 str_len = strlcpy((char *)(request.data), dev_name, (size_t)RMNET_MAX_STR_LEN);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600403 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
404 break;
405
406 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
407 != RMNETCTL_SUCCESS)
408 break;
409 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
410 break;
411 return_code = _rmnetctl_set_codes(response.return_code, error_code);
412 } while(0);
413 return return_code;
414}
415
416int rmnet_get_network_device_associated(rmnetctl_hndl_t *hndl,
417 const char *dev_name,
418 int *register_status,
419 uint16_t *error_code) {
420 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700421 size_t str_len = 0;
422 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600423 do {
424 if ((!hndl) || (!register_status) || (!error_code) ||
425 _rmnetctl_check_dev_name(dev_name)) {
426 return_code = RMNETCTL_INVALID_ARG;
427 break;
428 }
429
430 request.message_type = RMNET_NETLINK_GET_NETWORK_DEVICE_ASSOCIATED;
431
432 request.arg_length = RMNET_MAX_STR_LEN;
433 str_len = strlcpy((char *)(request.data), dev_name, RMNET_MAX_STR_LEN);
434 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
435 break;
436
437 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
438 != RMNETCTL_SUCCESS)
439 break;
440
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600441 if (_rmnetctl_check_data(response.crd, error_code)
442 != RMNETCTL_SUCCESS) {
443 if (_rmnetctl_check_code(response.crd, error_code)
444 == RMNETCTL_SUCCESS)
445 return_code = _rmnetctl_set_codes(response.return_code,
446 error_code);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600447 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600448 }
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600449
450 *register_status = response.return_code;
451 return_code = RMNETCTL_SUCCESS;
452 } while(0);
453 return return_code;
454}
455
456int rmnet_set_link_egress_data_format(rmnetctl_hndl_t *hndl,
457 uint32_t egress_flags,
458 uint16_t agg_size,
459 uint16_t agg_count,
460 const char *dev_name,
461 uint16_t *error_code) {
462 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700463 size_t str_len = 0;
464 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600465 do {
Harout Hedeshian77825572014-02-17 10:47:14 +0200466 if ((!hndl) || (!error_code) || _rmnetctl_check_dev_name(dev_name) ||
467 ((~EGRESS_FLAGS_MASK) & egress_flags)) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600468 return_code = RMNETCTL_INVALID_ARG;
469 break;
470 }
471
472 request.message_type = RMNET_NETLINK_SET_LINK_EGRESS_DATA_FORMAT;
473
474 request.arg_length = RMNET_MAX_STR_LEN +
475 sizeof(uint32_t) + sizeof(uint16_t) + sizeof(uint16_t);
476 str_len = strlcpy((char *)(request.data_format.dev),
477 dev_name,
478 RMNET_MAX_STR_LEN);
479 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
480 break;
481
482 request.data_format.flags = egress_flags;
483 request.data_format.agg_size = agg_size;
484 request.data_format.agg_count = agg_count;
485
486 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
487 != RMNETCTL_SUCCESS)
488 break;
489
490 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
491 break;
492
493 return_code = _rmnetctl_set_codes(response.return_code, error_code);
494 } while(0);
495 return return_code;
496}
497
498int rmnet_get_link_egress_data_format(rmnetctl_hndl_t *hndl,
499 const char *dev_name,
500 uint32_t *egress_flags,
501 uint16_t *agg_size,
502 uint16_t *agg_count,
503 uint16_t *error_code) {
504 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700505 size_t str_len = 0;
506 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600507 do {
508 if ((!hndl) || (!egress_flags) || (!agg_size) || (!agg_count) ||
509 (!error_code) || _rmnetctl_check_dev_name(dev_name)) {
510 return_code = RMNETCTL_INVALID_ARG;
511 break;
512 }
513 request.message_type = RMNET_NETLINK_GET_LINK_EGRESS_DATA_FORMAT;
514
515 request.arg_length = RMNET_MAX_STR_LEN;
516 str_len = strlcpy((char *)(request.data_format.dev),
517 dev_name,
518 RMNET_MAX_STR_LEN);
519 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
520 break;
521
522 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
523 != RMNETCTL_SUCCESS)
524 break;
525
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600526 if (_rmnetctl_check_data(response.crd, error_code)
527 != RMNETCTL_SUCCESS) {
528 if (_rmnetctl_check_code(response.crd, error_code)
529 == RMNETCTL_SUCCESS)
530 return_code = _rmnetctl_set_codes(response.return_code,
531 error_code);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600532 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600533 }
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600534
535 *egress_flags = response.data_format.flags;
536 *agg_size = response.data_format.agg_size;
537 *agg_count = response.data_format.agg_count;
538 return_code = RMNETCTL_SUCCESS;
539 } while(0);
540 return return_code;
541}
542
Subash Abhinov Kasiviswanathan22fa3882014-01-16 12:55:45 -0700543int rmnet_set_link_ingress_data_format_tailspace(rmnetctl_hndl_t *hndl,
544 uint32_t ingress_flags,
545 uint8_t tail_spacing,
546 const char *dev_name,
547 uint16_t *error_code) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600548 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700549 size_t str_len = 0;
550 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600551 do {
Harout Hedeshian77825572014-02-17 10:47:14 +0200552 if ((!hndl) || (!error_code) || _rmnetctl_check_dev_name(dev_name) ||
553 ((~INGRESS_FLAGS_MASK) & ingress_flags)) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600554 return_code = RMNETCTL_INVALID_ARG;
555 break;
556 }
557
558 request.message_type = RMNET_NETLINK_SET_LINK_INGRESS_DATA_FORMAT;
559
560 request.arg_length = RMNET_MAX_STR_LEN +
561 sizeof(uint32_t) + sizeof(uint16_t) + sizeof(uint16_t);
562 str_len = strlcpy((char *)(request.data_format.dev),
563 dev_name,
564 RMNET_MAX_STR_LEN);
565 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
566 break;
567 request.data_format.flags = ingress_flags;
Subash Abhinov Kasiviswanathan22fa3882014-01-16 12:55:45 -0700568 request.data_format.tail_spacing = tail_spacing;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600569
570 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
571 != RMNETCTL_SUCCESS)
572 break;
573
574 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
575 break;
576
577 return_code = _rmnetctl_set_codes(response.return_code, error_code);
578 } while(0);
579 return return_code;
580}
581
Subash Abhinov Kasiviswanathan22fa3882014-01-16 12:55:45 -0700582int rmnet_get_link_ingress_data_format_tailspace(rmnetctl_hndl_t *hndl,
583 const char *dev_name,
584 uint32_t *ingress_flags,
585 uint8_t *tail_spacing,
586 uint16_t *error_code) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600587 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700588 size_t str_len = 0;
589 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600590 do {
Subash Abhinov Kasiviswanathan22fa3882014-01-16 12:55:45 -0700591 if ((!hndl) || (!error_code) ||
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600592 _rmnetctl_check_dev_name(dev_name)) {
593 return_code = RMNETCTL_INVALID_ARG;
594 break;
595 }
596
597 request.message_type = RMNET_NETLINK_GET_LINK_INGRESS_DATA_FORMAT;
598
599 request.arg_length = RMNET_MAX_STR_LEN;
600 str_len = strlcpy((char *)(request.data_format.dev),
601 dev_name,
602 RMNET_MAX_STR_LEN);
603 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
604 break;
605
606 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
607 != RMNETCTL_SUCCESS)
608 break;
609
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600610 if (_rmnetctl_check_data(response.crd, error_code)
611 != RMNETCTL_SUCCESS) {
612 if (_rmnetctl_check_code(response.crd, error_code)
613 == RMNETCTL_SUCCESS)
614 return_code = _rmnetctl_set_codes(response.return_code,
615 error_code);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600616 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600617 }
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600618
Subash Abhinov Kasiviswanathan22fa3882014-01-16 12:55:45 -0700619 if (ingress_flags)
620 *ingress_flags = response.data_format.flags;
621
622 if (tail_spacing)
623 *tail_spacing = response.data_format.tail_spacing;
624
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600625 return_code = RMNETCTL_SUCCESS;
626 } while(0);
627 return return_code;
628}
629
630int rmnet_set_logical_ep_config(rmnetctl_hndl_t *hndl,
631 int32_t ep_id,
632 uint8_t operating_mode,
633 const char *dev_name,
634 const char *next_dev,
635 uint16_t *error_code) {
636 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700637 size_t str_len = 0;
638 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600639 do {
640 if ((!hndl) || ((ep_id < -1) || (ep_id > 31)) || (!error_code) ||
641 _rmnetctl_check_dev_name(dev_name) ||
Harout Hedeshian77825572014-02-17 10:47:14 +0200642 _rmnetctl_check_dev_name(next_dev) ||
643 operating_mode >= RMNET_EPMODE_LENGTH) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600644 return_code = RMNETCTL_INVALID_ARG;
645 break;
646 }
647
648 request.message_type = RMNET_NETLINK_SET_LOGICAL_EP_CONFIG;
649
650 request.arg_length = RMNET_MAX_STR_LEN +
651 RMNET_MAX_STR_LEN + sizeof(int32_t) + sizeof(uint8_t);
652 str_len = strlcpy((char *)(request.local_ep_config.dev),
653 dev_name,
654 RMNET_MAX_STR_LEN);
655 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
656 break;
657
658 str_len = strlcpy((char *)(request.local_ep_config.next_dev),
659 next_dev,
660 RMNET_MAX_STR_LEN);
661 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
662 break;
663 request.local_ep_config.ep_id = ep_id;
664 request.local_ep_config.operating_mode = operating_mode;
665
666 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
667 != RMNETCTL_SUCCESS)
668 break;
669 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
670 break;
671
672 return_code = _rmnetctl_set_codes(response.return_code, error_code);
673 } while(0);
674 return return_code;
675}
676
Harout Hedeshiana9731652014-01-06 18:00:23 +0200677int rmnet_unset_logical_ep_config(rmnetctl_hndl_t *hndl,
678 int32_t ep_id,
679 const char *dev_name,
680 uint16_t *error_code) {
681 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700682 size_t str_len = 0;
683 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshiana9731652014-01-06 18:00:23 +0200684 do {
685
686 if ((!hndl) || ((ep_id < -1) || (ep_id > 31)) || (!error_code) ||
687 _rmnetctl_check_dev_name(dev_name)) {
688 return_code = RMNETCTL_INVALID_ARG;
689 break;
690 }
691
692 request.message_type = RMNET_NETLINK_UNSET_LOGICAL_EP_CONFIG;
693
694 request.arg_length = RMNET_MAX_STR_LEN + sizeof(int32_t);
695 str_len = strlcpy((char *)(request.local_ep_config.dev),
696 dev_name,
697 RMNET_MAX_STR_LEN);
698
699 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
700 break;
701
702 request.local_ep_config.ep_id = ep_id;
703
704 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
705 != RMNETCTL_SUCCESS)
706 break;
707 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
708 break;
709
710 return_code = _rmnetctl_set_codes(response.return_code, error_code);
711 } while(0);
712
713 return return_code;
714}
715
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600716int rmnet_get_logical_ep_config(rmnetctl_hndl_t *hndl,
717 int32_t ep_id,
718 const char *dev_name,
719 uint8_t *operating_mode,
720 char **next_dev,
Harout Hedeshian77825572014-02-17 10:47:14 +0200721 uint32_t next_dev_len,
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600722 uint16_t *error_code) {
723 struct rmnet_nl_msg_s request, response;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700724 size_t str_len = 0;
725 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600726 do {
727 if ((!hndl) || (!operating_mode) || (!error_code) || ((ep_id < -1) ||
Harout Hedeshian77825572014-02-17 10:47:14 +0200728 (ep_id > 31)) || _rmnetctl_check_dev_name(dev_name) || (!next_dev)
729 || (0 == next_dev_len)) {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600730 return_code = RMNETCTL_INVALID_ARG;
731 break;
732 }
733
734 request.message_type = RMNET_NETLINK_GET_LOGICAL_EP_CONFIG;
735
736 request.arg_length = RMNET_MAX_STR_LEN + sizeof(int32_t);
737 str_len = strlcpy((char *)(request.local_ep_config.dev),
738 dev_name,
739 RMNET_MAX_STR_LEN);
740 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
741 break;
742
743 request.local_ep_config.ep_id = ep_id;
744
745 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
746 != RMNETCTL_SUCCESS)
747 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600748
749 if (_rmnetctl_check_data(response.crd, error_code)
750 != RMNETCTL_SUCCESS) {
751 if (_rmnetctl_check_code(response.crd, error_code)
752 == RMNETCTL_SUCCESS)
753 return_code = _rmnetctl_set_codes(response.return_code,
754 error_code);
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600755 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600756 }
Harout Hedeshian77825572014-02-17 10:47:14 +0200757
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600758 str_len = strlcpy(*next_dev,
759 (char *)(response.local_ep_config.next_dev),
Harout Hedeshian77825572014-02-17 10:47:14 +0200760 min(RMNET_MAX_STR_LEN, next_dev_len));
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600761 if (_rmnetctl_check_len(str_len, error_code) != RMNETCTL_SUCCESS)
762 break;
763
764 *operating_mode = response.local_ep_config.operating_mode;
765 return_code = RMNETCTL_SUCCESS;
766 } while(0);
767 return return_code;
768}
769
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600770int rmnet_new_vnd_prefix(rmnetctl_hndl_t *hndl,
771 uint32_t id,
772 uint16_t *error_code,
773 uint8_t new_vnd,
774 const char *prefix)
775{
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600776 struct rmnet_nl_msg_s request, response;
777 int return_code = RMNETCTL_LIB_ERR;
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700778 size_t str_len = 0;
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600779 do {
780 if ((!hndl) || (!error_code) ||
781 ((new_vnd != RMNETCTL_NEW_VND) && (new_vnd != RMNETCTL_FREE_VND))) {
782 return_code = RMNETCTL_INVALID_ARG;
783 break;
784 }
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600785
786 memset(request.vnd.vnd_name, 0, RMNET_MAX_STR_LEN);
787 if (new_vnd == RMNETCTL_NEW_VND) {
788 if (prefix) {
789 request.message_type =RMNET_NETLINK_NEW_VND_WITH_PREFIX;
790 str_len = strlcpy((char *)request.vnd.vnd_name,
791 prefix, RMNET_MAX_STR_LEN);
792 if (_rmnetctl_check_len(str_len, error_code)
793 != RMNETCTL_SUCCESS)
794 break;
795 } else {
796 request.message_type = RMNET_NETLINK_NEW_VND;
797 }
798 } else {
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600799 request.message_type = RMNET_NETLINK_FREE_VND;
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600800 }
Harout Hedeshianb2fc5b12013-09-03 13:49:00 -0600801
802 request.arg_length = sizeof(uint32_t);
803 request.vnd.id = id;
804
805 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
806 != RMNETCTL_SUCCESS)
807 break;
808 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
809 break;
810
811 return_code = _rmnetctl_set_codes(response.return_code, error_code);
812 } while(0);
813 return return_code;
814}
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600815
816int rmnet_new_vnd(rmnetctl_hndl_t *hndl,
817 uint32_t id,
818 uint16_t *error_code,
819 uint8_t new_vnd)
820{
821 return rmnet_new_vnd_prefix(hndl, id, error_code, new_vnd, 0);
822}
823
824int rmnet_get_vnd_name(rmnetctl_hndl_t *hndl,
825 uint32_t id,
826 uint16_t *error_code,
827 char *buf,
828 uint32_t buflen)
829{
830 struct rmnet_nl_msg_s request, response;
831 uint32_t str_len;
832 int return_code = RMNETCTL_LIB_ERR;
833 do {
Harout Hedeshian77825572014-02-17 10:47:14 +0200834 if ((!hndl) || (!error_code) || (!buf) || (0 == buflen)) {
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600835 return_code = RMNETCTL_INVALID_ARG;
836 break;
837 }
838
839 request.message_type = RMNET_NETLINK_GET_VND_NAME;
840 request.arg_length = sizeof(uint32_t);
841 request.vnd.id = id;
842
843
844 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
845 != RMNETCTL_SUCCESS)
846 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600847
848 if (_rmnetctl_check_data(response.crd, error_code)
849 != RMNETCTL_SUCCESS) {
850 if (_rmnetctl_check_code(response.crd, error_code)
851 == RMNETCTL_SUCCESS)
852 return_code = _rmnetctl_set_codes(response.return_code,
853 error_code);
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600854 break;
Subash Abhinov Kasiviswanathand42712d2015-04-14 13:44:02 -0600855 }
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600856
Harout Hedeshian0bf0a172014-02-19 23:52:05 -0700857 str_len = (uint32_t)strlcpy(buf,
Harout Hedeshian89a91e22013-10-09 08:59:47 -0600858 (char *)(response.vnd.vnd_name),
859 buflen);
860 if (str_len >= buflen) {
861 *error_code = RMNETCTL_API_ERR_STRING_TRUNCATION;
862 break;
863 }
864
865 return_code = RMNETCTL_SUCCESS;
866 } while (0);
867 return return_code;
868}
869
Harout Hedeshian97a1e982013-11-08 09:28:53 -0700870int rmnet_add_del_vnd_tc_flow(rmnetctl_hndl_t *hndl,
871 uint32_t id,
872 uint32_t map_flow_id,
873 uint32_t tc_flow_id,
874 uint8_t set_flow,
875 uint16_t *error_code) {
876 struct rmnet_nl_msg_s request, response;
877 int return_code = RMNETCTL_LIB_ERR;
878 do {
Harout Hedeshian77825572014-02-17 10:47:14 +0200879 if ((!hndl) || (!error_code) || ((set_flow != RMNETCTL_ADD_FLOW) &&
880 (set_flow != RMNETCTL_DEL_FLOW))) {
Harout Hedeshian97a1e982013-11-08 09:28:53 -0700881 return_code = RMNETCTL_INVALID_ARG;
882 break;
883 }
884 if (set_flow == RMNETCTL_ADD_FLOW)
885 request.message_type = RMNET_NETLINK_ADD_VND_TC_FLOW;
886 else
887 request.message_type = RMNET_NETLINK_DEL_VND_TC_FLOW;
888
889 request.arg_length = (sizeof(uint32_t))*3;
890 request.flow_control.id = id;
891 request.flow_control.map_flow_id = map_flow_id;
892 request.flow_control.tc_flow_id = tc_flow_id;
893
894 if ((*error_code = rmnetctl_transact(hndl, &request, &response))
895 != RMNETCTL_SUCCESS)
896 break;
897 if (_rmnetctl_check_code(response.crd, error_code) != RMNETCTL_SUCCESS)
898 break;
899 return_code = _rmnetctl_set_codes(response.return_code, error_code);
900 } while(0);
901 return return_code;
902}
903