blob: 74db1845080b436c80e069056ad16e1d7be596ef [file] [log] [blame]
/*
* Copyright (C) 2015 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.
*/
#include <android/multinetwork.h>
#include <errno.h>
#include <NetdClient.h> // the functions that communicate with netd
#include <resolv_netid.h> // android_getaddrinfofornet()
#include <stdlib.h>
#include <sys/limits.h>
// This value MUST be kept in sync with the corresponding value in
// the android.net.Network#getNetworkHandle() implementation.
static const uint32_t kHandleMagic = 0xcafed00d;
static const uint32_t kHandleMagicSize = 32;
static int getnetidfromhandle(net_handle_t handle, unsigned *netid) {
static const uint32_t k32BitMask = 0xffffffff;
// Check for minimum acceptable version of the API in the low bits.
if (handle != NETWORK_UNSPECIFIED &&
(handle & k32BitMask) != kHandleMagic) {
return 0;
}
if (netid != NULL) {
*netid = ((handle >> (CHAR_BIT * sizeof(k32BitMask))) & k32BitMask);
}
return 1;
}
static net_handle_t gethandlefromnetid(unsigned netid) {
if (netid == NETID_UNSET) {
return NETWORK_UNSPECIFIED;
}
return (((net_handle_t) netid) << kHandleMagicSize) | kHandleMagic;
}
int android_setsocknetwork(net_handle_t network, int fd) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
errno = EINVAL;
return -1;
}
int rval = setNetworkForSocket(netid, fd);
if (rval < 0) {
errno = -rval;
rval = -1;
}
return rval;
}
int android_setprocnetwork(net_handle_t network) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
errno = EINVAL;
return -1;
}
int rval = setNetworkForProcess(netid);
if (rval < 0) {
errno = -rval;
rval = -1;
}
return rval;
}
int android_getprocnetwork(net_handle_t *network) {
if (network == NULL) {
errno = EINVAL;
return -1;
}
unsigned netid = getNetworkForProcess();
*network = gethandlefromnetid(netid);
return 0;
}
int android_setprocdns(net_handle_t network) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
errno = EINVAL;
return -1;
}
int rval = setNetworkForResolv(netid);
if (rval < 0) {
errno = -rval;
rval = -1;
}
return rval;
}
int android_getprocdns(net_handle_t *network) {
if (network == NULL) {
errno = EINVAL;
return -1;
}
unsigned netid;
int rval = getNetworkForDns(&netid);
if (rval < 0) {
errno = -rval;
return -1;
}
*network = gethandlefromnetid(netid);
return 0;
}
int android_getaddrinfofornetwork(net_handle_t network,
const char *node, const char *service,
const struct addrinfo *hints, struct addrinfo **res) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
errno = EINVAL;
return EAI_SYSTEM;
}
return android_getaddrinfofornet(node, service, hints, netid, 0, res);
}
int android_res_nquery(net_handle_t network, const char *dname,
int ns_class, int ns_type, enum ResNsendFlags flags) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
return -ENONET;
}
return resNetworkQuery(netid, dname, ns_class, ns_type, flags);
}
int android_res_nresult(int fd, int *rcode, uint8_t *answer, size_t anslen) {
return resNetworkResult(fd, rcode, answer, anslen);
}
int android_res_nsend(net_handle_t network, const uint8_t *msg, size_t msglen,
enum ResNsendFlags flags) {
unsigned netid;
if (!getnetidfromhandle(network, &netid)) {
return -ENONET;
}
return resNetworkSend(netid, msg, msglen, flags);
}
void android_res_cancel(int nsend_fd) {
resNetworkCancel(nsend_fd);
}
int android_tag_socket_with_uid(int sockfd, uint32_t tag, uid_t uid) {
return tagSocket(sockfd, tag, uid);
}
int android_tag_socket(int sockfd, uint32_t tag) {
return tagSocket(sockfd, tag, -1);
}
int android_untag_socket(int sockfd) {
return untagSocket(sockfd);
}