/* * Copyright (C) 2011 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* NOTICE: This is a clean room re-implementation of libnl */ #include <errno.h> #include <unistd.h> #include <malloc.h> #include <sys/time.h> #include <sys/socket.h> #include "netlink-types.h" /* Join group */ int nl_socket_add_membership(struct nl_sock *sk, int group) { return setsockopt(sk->s_fd, SOL_NETLINK, NETLINK_ADD_MEMBERSHIP, &group, sizeof(group)); } /* Allocate new netlink socket. */ static struct nl_sock *_nl_socket_alloc(void) { struct nl_sock *sk; struct timeval tv; struct nl_cb *cb; sk = (struct nl_sock *) malloc(sizeof(struct nl_sock)); if (!sk) return NULL; memset(sk, 0, sizeof(*sk)); /* Get current time */ if (gettimeofday(&tv, NULL)) goto fail; else sk->s_seq_next = (int) tv.tv_sec; /* Create local socket */ sk->s_local.nl_family = AF_NETLINK; sk->s_local.nl_pid = 0; /* Kernel fills in pid */ sk->s_local.nl_groups = 0; /* No groups */ /* Create peer socket */ sk->s_peer.nl_family = AF_NETLINK; sk->s_peer.nl_pid = 0; /* Kernel */ sk->s_peer.nl_groups = 0; /* No groups */ return sk; fail: free(sk); return NULL; } /* Allocate new netlink socket. */ struct nl_sock *nl_socket_alloc(void) { struct nl_sock *sk = _nl_socket_alloc(); struct nl_cb *cb; if (!sk) return NULL; cb = nl_cb_alloc(NL_CB_DEFAULT); if (!cb) goto cb_fail; sk->s_cb = cb; return sk; cb_fail: free(sk); return NULL; } /* Allocate new socket with custom callbacks. */ struct nl_sock *nl_socket_alloc_cb(struct nl_cb *cb) { struct nl_sock *sk = _nl_socket_alloc(); if (!sk) return NULL; sk->s_cb = cb; nl_cb_get(cb); return sk; } /* Free a netlink socket. */ void nl_socket_free(struct nl_sock *sk) { nl_cb_put(sk->s_cb); close(sk->s_fd); free(sk); } /* Sets socket buffer size of netlink socket */ int nl_socket_set_buffer_size(struct nl_sock *sk, int rxbuf, int txbuf) { if (setsockopt(sk->s_fd, SOL_SOCKET, SO_SNDBUF, \ &rxbuf, (socklen_t) sizeof(rxbuf))) goto error; if (setsockopt(sk->s_fd, SOL_SOCKET, SO_RCVBUF, \ &txbuf, (socklen_t) sizeof(txbuf))) goto error; return 0; error: return -errno; } int nl_socket_get_fd(struct nl_sock *sk) { return sk->s_fd; } void nl_socket_set_cb(struct nl_sock *sk, struct nl_cb *cb) { nl_cb_put(sk->s_cb); sk->s_cb = cb; nl_cb_get(cb); } struct nl_cb *nl_socket_get_cb(struct nl_sock *sk) { return nl_cb_get(sk->s_cb); }