|  | /* Copyright (c) 2017, The Linux Foundation. All rights reserved. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions are | 
|  | * met: | 
|  | *     * Redistributions of source code must retain the above copyright | 
|  | *       notice, this list of conditions and the following disclaimer. | 
|  | *     * Redistributions in binary form must reproduce the above | 
|  | *       copyright notice, this list of conditions and the following | 
|  | *       disclaimer in the documentation and/or other materials provided | 
|  | *       with the distribution. | 
|  | *     * Neither the name of The Linux Foundation nor the names of its | 
|  | *       contributors may be used to endorse or promote products derived | 
|  | *       from this software without specific prior written permission. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED | 
|  | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF | 
|  | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT | 
|  | * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS | 
|  | * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | 
|  | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | 
|  | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR | 
|  | * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | 
|  | * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE | 
|  | * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN | 
|  | * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | * | 
|  | */ | 
|  |  | 
|  | #ifndef CLD80211_LIB_H | 
|  | #define CLD80211_LIB_H | 
|  |  | 
|  | #ifdef __cplusplus | 
|  | extern "C" { | 
|  | #endif | 
|  | #include <netlink/genl/genl.h> | 
|  | #include <stdbool.h> | 
|  |  | 
|  | #ifndef UNUSED | 
|  | #define UNUSED(x)    (void)(x) | 
|  | #endif | 
|  |  | 
|  | struct cld80211_ctx { | 
|  | struct nl_sock *sock; | 
|  | int netlink_familyid; | 
|  | /* socket pair used to exit from blocking poll*/ | 
|  | int exit_sockets[2]; | 
|  | int sock_buf_size; | 
|  | int nlctrl_familyid; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * enum cld80211_attr - Driver/Application embeds the data in nlmsg with the | 
|  | *                      help of below attributes | 
|  | * CLD80211_ATTR_VENDOR_DATA: Embed all other attributes in this nested | 
|  | *                            attribute. | 
|  | * CLD80211_ATTR_DATA: Embed driver/application data in this attribute | 
|  | * CLD80211_ATTR_META_DATA: Embed meta data for above data. This will  help | 
|  | * wlan driver to peek into request message packet without opening up definition | 
|  | * of complete request message. | 
|  | * | 
|  | * Any new message in future can be added as another attribute | 
|  | */ | 
|  | enum cld80211_attr { | 
|  | CLD80211_ATTR_VENDOR_DATA = 1, | 
|  | CLD80211_ATTR_DATA, | 
|  | CLD80211_ATTR_META_DATA, | 
|  |  | 
|  | __CLD80211_ATTR_AFTER_LAST, | 
|  | CLD80211_ATTR_MAX = __CLD80211_ATTR_AFTER_LAST - 1 | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Create socket of type NETLINK_GENERIC | 
|  | * Retuns valid sock only if socket creation is succesful and cld80211 | 
|  | * family is present, returns NULL otherwise | 
|  | */ | 
|  | struct cld80211_ctx *cld80211_init(); | 
|  |  | 
|  | /** | 
|  | * free the socket created in cld80211_init() | 
|  | */ | 
|  | void cld80211_deinit(struct cld80211_ctx *ctx); | 
|  |  | 
|  | /** | 
|  | * Allocate nl_msg and populate family and genl header details | 
|  | */ | 
|  | struct nl_msg *cld80211_msg_alloc(struct cld80211_ctx *ctx, int cmd, | 
|  | struct nlattr **nla_data, int pid); | 
|  |  | 
|  | /** | 
|  | * Send nlmsg to driver and return; It doesn't wait for response | 
|  | */ | 
|  | int cld80211_send_msg(struct cld80211_ctx *ctx, struct nl_msg *nlmsg); | 
|  |  | 
|  | /** | 
|  | * Send nlmsg to driver and get response, if any | 
|  | */ | 
|  | int cld80211_send_recv_msg(struct cld80211_ctx *ctx, struct nl_msg *nlmsg, | 
|  | int (*valid_handler)(struct nl_msg *, void *), | 
|  | void *valid_data); | 
|  |  | 
|  | /** | 
|  | * Add membership for multicast group "mcgroup" to receive the messages | 
|  | * sent to this group from driver | 
|  | */ | 
|  | int cld80211_add_mcast_group(struct cld80211_ctx *ctx, const char* mcgroup); | 
|  |  | 
|  | /** | 
|  | * Remove membership of multicast group "mcgroup" to stop receiving messages | 
|  | * sent to this group from driver | 
|  | */ | 
|  | int cld80211_remove_mcast_group(struct cld80211_ctx *ctx, const char* mcgroup); | 
|  |  | 
|  | /** | 
|  | * Receive messages from driver on cld80211 family. Client can do | 
|  | * a select()/poll() on the socket before calling this API. | 
|  | * sock: nl_sock created for communication | 
|  | * cb: nl callback context provided by client | 
|  | * Returns corresponding errno when a failure happens while receiving nl msg | 
|  | */ | 
|  | int cld80211_recv_msg(struct nl_sock *sock, struct nl_cb *cb); | 
|  |  | 
|  | /** | 
|  | * Receive messages from driver on cld80211 family from the | 
|  | * multicast groups subscribed | 
|  | * timeout: Timeout in milliseconds for poll(); -1 is for infinite timeout. | 
|  | * recv_multi_msg: Boolean flag to be sent false/true from client to indicate | 
|  | *                 whether it wants to receive only one message or multiple | 
|  | *                 messages from timeoutblock. | 
|  | *                 false: Receive only one message and return | 
|  | *                 true: Continue in the loop to receive multiple message till | 
|  | *                       client explicitly sends exit via exit_cld80211_recv(). | 
|  | * cbctx: Context provided by client, which is to be used when an | 
|  | *        nlmsg is received | 
|  | * Returns corresponding errno when a failure happens while receiving nl msg | 
|  | */ | 
|  | int cld80211_recv(struct cld80211_ctx *ctx, int timeout, bool recv_multi_msg, | 
|  | int (*valid_handler)(struct nl_msg *, void *), | 
|  | void *cbctx); | 
|  |  | 
|  | /** | 
|  | * poll() is a blocking call on sock. Client has to unblock the poll() | 
|  | * first to exit gracefully. | 
|  | */ | 
|  | void exit_cld80211_recv(struct cld80211_ctx *ctx); | 
|  | #ifdef __cplusplus | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #endif |