/* * 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 #include #include #include "netlink-types.h" /* Allocate a new netlink message with the default maximum payload size. */ struct nl_msg *nlmsg_alloc(void) { /* Whole page will store nl_msg + nlmsghdr + genlmsghdr + payload */ const int page_sz = getpagesize(); struct nl_msg *nm; struct nlmsghdr *nlh; /* Netlink message */ nm = (struct nl_msg *) malloc(page_sz); if (!nm) goto fail; /* Netlink message header pointer */ nlh = (struct nlmsghdr *) ((char *) nm + sizeof(struct nl_msg)); /* Initialize */ memset(nm, 0, page_sz); nm->nm_size = page_sz; nm->nm_src.nl_family = AF_NETLINK; nm->nm_src.nl_pid = getpid(); nm->nm_dst.nl_family = AF_NETLINK; nm->nm_dst.nl_pid = 0; /* Kernel */ /* Initialize and add to netlink message */ nlh->nlmsg_len = NLMSG_HDRLEN; nm->nm_nlh = nlh; /* Add to reference count and return nl_msg */ nlmsg_get(nm); return nm; fail: return NULL; } /* Return pointer to message payload. */ void *nlmsg_data(const struct nlmsghdr *nlh) { return (char *) nlh + NLMSG_HDRLEN; } /* Add reference count to nl_msg */ void nlmsg_get(struct nl_msg *nm) { nm->nm_refcnt++; } /* Release a reference from an netlink message. */ void nlmsg_free(struct nl_msg *nm) { if (nm) { nm->nm_refcnt--; if (nm->nm_refcnt <= 0) free(nm); } } /* Return actual netlink message. */ struct nlmsghdr *nlmsg_hdr(struct nl_msg *n) { return n->nm_nlh; } /* Return head of attributes data / payload section */ struct nlattr *nlmsg_attrdata(const struct nlmsghdr *nlh, int hdrlen) { unsigned char *data = nlmsg_data(nlh); return (struct nlattr *)(data + NLMSG_ALIGN(hdrlen)); } /* Returns pointer to end of netlink message */ void *nlmsg_tail(const struct nlmsghdr *nlh) { return (void *)((char *)nlh + NLMSG_ALIGN(nlh->nlmsg_len)); } /* Next netlink message in message stream */ struct nlmsghdr *nlmsg_next(struct nlmsghdr *nlh, int *remaining) { struct nlmsghdr *next_nlh = NULL; int len = nlmsg_len(nlh); len = NLMSG_ALIGN(len); if (*remaining > 0 && len <= *remaining && len >= (int) sizeof(struct nlmsghdr)) { next_nlh = (struct nlmsghdr *)((char *)nlh + len); *remaining -= len; } return next_nlh; } int nlmsg_datalen(const struct nlmsghdr *nlh) { return nlh->nlmsg_len - NLMSG_HDRLEN; } /* Length of attributes data */ int nlmsg_attrlen(const struct nlmsghdr *nlh, int hdrlen) { return nlmsg_datalen(nlh) - NLMSG_ALIGN(hdrlen); } /* Length of netlink message */ int nlmsg_len(const struct nlmsghdr *nlh) { return nlh->nlmsg_len; } /* Check if the netlink message fits into the remaining bytes */ int nlmsg_ok(const struct nlmsghdr *nlh, int rem) { return rem >= (int)sizeof(struct nlmsghdr) && rem >= nlmsg_len(nlh) && nlmsg_len(nlh) >= (int) sizeof(struct nlmsghdr) && nlmsg_len(nlh) <= (rem); } int nlmsg_padlen(int payload) { return NLMSG_ALIGN(payload) - payload; }