Merge commit '811b0cdb2d6e4a697dbc63a678712759dd0db242' into HEAD
Change-Id: I786944f80fb1a2d502fed51dc2c391ed5db66761
This commit is contained in:
commit
f00c938c7f
@ -102,6 +102,7 @@ libc_common_src_files := \
|
|||||||
bionic/md5.c \
|
bionic/md5.c \
|
||||||
bionic/memmem.c \
|
bionic/memmem.c \
|
||||||
bionic/memswap.c \
|
bionic/memswap.c \
|
||||||
|
bionic/name_mem.c \
|
||||||
bionic/pathconf.c \
|
bionic/pathconf.c \
|
||||||
bionic/perror.c \
|
bionic/perror.c \
|
||||||
bionic/ptsname.c \
|
bionic/ptsname.c \
|
||||||
|
@ -73,8 +73,10 @@ ENTRY_PRIVATE(MEMCPY_BASE)
|
|||||||
cmp r2, #16
|
cmp r2, #16
|
||||||
blo .L_copy_less_than_16_unknown_align
|
blo .L_copy_less_than_16_unknown_align
|
||||||
|
|
||||||
cmp r2, #832
|
// TODO: The aligned copy code is extremely slow copying some large
|
||||||
bge .L_check_alignment
|
// buffers so always go through the unaligned path for now.
|
||||||
|
//cmp r2, #832
|
||||||
|
//bge .L_check_alignment
|
||||||
|
|
||||||
.L_copy_unknown_alignment:
|
.L_copy_unknown_alignment:
|
||||||
// Unknown alignment of src and dst.
|
// Unknown alignment of src and dst.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2008 The Android Open Source Project
|
* Copyright (C) 2013 The Android Open Source Project
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
#include "dlmalloc.h"
|
#include "dlmalloc.h"
|
||||||
|
|
||||||
|
#include "private/bionic_name_mem.h"
|
||||||
#include "private/libc_logging.h"
|
#include "private/libc_logging.h"
|
||||||
|
|
||||||
// Send dlmalloc errors to the log.
|
// Send dlmalloc errors to the log.
|
||||||
@ -25,6 +26,11 @@ static void __bionic_heap_usage_error(const char* function, void* address);
|
|||||||
#define CORRUPTION_ERROR_ACTION(m) __bionic_heap_corruption_error(__FUNCTION__)
|
#define CORRUPTION_ERROR_ACTION(m) __bionic_heap_corruption_error(__FUNCTION__)
|
||||||
#define USAGE_ERROR_ACTION(m,p) __bionic_heap_usage_error(__FUNCTION__, p)
|
#define USAGE_ERROR_ACTION(m,p) __bionic_heap_usage_error(__FUNCTION__, p)
|
||||||
|
|
||||||
|
/* Bionic named anonymous memory declarations */
|
||||||
|
static void* named_anonymous_mmap(size_t length);
|
||||||
|
#define MMAP(s) named_anonymous_mmap(s)
|
||||||
|
#define DIRECT_MMAP(s) named_anonymous_mmap(s)
|
||||||
|
|
||||||
// Ugly inclusion of C file so that bionic specific #defines configure dlmalloc.
|
// Ugly inclusion of C file so that bionic specific #defines configure dlmalloc.
|
||||||
#include "../upstream-dlmalloc/malloc.c"
|
#include "../upstream-dlmalloc/malloc.c"
|
||||||
|
|
||||||
@ -42,3 +48,15 @@ static void __bionic_heap_usage_error(const char* function, void* address) {
|
|||||||
// TODO: improve the debuggerd protocol so we can tell it to dump an address when we abort.
|
// TODO: improve the debuggerd protocol so we can tell it to dump an address when we abort.
|
||||||
*((int**) 0xdeadbaad) = (int*) address;
|
*((int**) 0xdeadbaad) = (int*) address;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void* named_anonymous_mmap(size_t length)
|
||||||
|
{
|
||||||
|
void* ret;
|
||||||
|
ret = mmap(NULL, length, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
|
||||||
|
if (ret == MAP_FAILED)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
__bionic_name_mem(ret, length, "libc_malloc");
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
53
libc/bionic/name_mem.c
Normal file
53
libc/bionic/name_mem.c
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2013 The Android Open Source Project
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "private/bionic_name_mem.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Local definitions of custom prctl arguments to set a vma name in some kernels
|
||||||
|
*/
|
||||||
|
#define BIONIC_PR_SET_VMA 0x53564d41
|
||||||
|
#define BIONIC_PR_SET_VMA_ANON_NAME 0
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Names a region of memory. The name is expected to show up in /proc/pid/maps
|
||||||
|
* and /proc/pid/smaps. There is no guarantee that it will work, and it if it
|
||||||
|
* does work it is likely to only work on memory that was allocated with
|
||||||
|
* mmap(MAP_ANONYMOUS), and only on regions that are page aligned. name should
|
||||||
|
* be a pointer to a string that is valid for as long as the memory is mapped,
|
||||||
|
* preferably a compile-time constant string.
|
||||||
|
*
|
||||||
|
* Returns -1 on error and sets errno. If it returns an error naming page
|
||||||
|
* aligned anonymous memory the kernel doesn't support naming, and an alternate
|
||||||
|
* method of naming memory should be used (like ashmem).
|
||||||
|
*/
|
||||||
|
int __bionic_name_mem(void *addr, size_t len, const char *name)
|
||||||
|
{
|
||||||
|
return prctl(BIONIC_PR_SET_VMA, BIONIC_PR_SET_VMA_ANON_NAME,
|
||||||
|
addr, len, name);
|
||||||
|
}
|
@ -207,13 +207,13 @@ void endprotoent(void);
|
|||||||
void endservent(void);
|
void endservent(void);
|
||||||
void freehostent(struct hostent *);
|
void freehostent(struct hostent *);
|
||||||
struct hostent *gethostbyaddr(const void *, socklen_t, int);
|
struct hostent *gethostbyaddr(const void *, socklen_t, int);
|
||||||
struct hostent *android_gethostbyaddrforiface(const void *, socklen_t, int, const char*);
|
struct hostent *android_gethostbyaddrforiface(const void *, socklen_t, int, const char*, int);
|
||||||
int gethostbyaddr_r(const void *, int, int, struct hostent *, char *, size_t, struct hostent **, int *);
|
int gethostbyaddr_r(const void *, int, int, struct hostent *, char *, size_t, struct hostent **, int *);
|
||||||
struct hostent *gethostbyname(const char *);
|
struct hostent *gethostbyname(const char *);
|
||||||
int gethostbyname_r(const char *, struct hostent *, char *, size_t, struct hostent **, int *);
|
int gethostbyname_r(const char *, struct hostent *, char *, size_t, struct hostent **, int *);
|
||||||
struct hostent *gethostbyname2(const char *, int);
|
struct hostent *gethostbyname2(const char *, int);
|
||||||
int gethostbyname2_r(const char *, int, struct hostent *, char *, size_t, struct hostent **, int *);
|
int gethostbyname2_r(const char *, int, struct hostent *, char *, size_t, struct hostent **, int *);
|
||||||
struct hostent *android_gethostbynameforiface(const char *, int, const char *);
|
struct hostent *android_gethostbynameforiface(const char *, int, const char *, int);
|
||||||
struct hostent *gethostent(void);
|
struct hostent *gethostent(void);
|
||||||
int gethostent_r(struct hostent *, char *, size_t, struct hostent **, int *);
|
int gethostent_r(struct hostent *, char *, size_t, struct hostent **, int *);
|
||||||
struct hostent *getipnodebyaddr(const void *, size_t, int, int *);
|
struct hostent *getipnodebyaddr(const void *, size_t, int, int *);
|
||||||
@ -241,9 +241,9 @@ void sethostent(int);
|
|||||||
void setnetent(int);
|
void setnetent(int);
|
||||||
void setprotoent(int);
|
void setprotoent(int);
|
||||||
int getaddrinfo(const char *, const char *, const struct addrinfo *, struct addrinfo **);
|
int getaddrinfo(const char *, const char *, const struct addrinfo *, struct addrinfo **);
|
||||||
int android_getaddrinfoforiface(const char *, const char *, const struct addrinfo *, const char *, struct addrinfo **);
|
int android_getaddrinfoforiface(const char *, const char *, const struct addrinfo *, const char *, int, struct addrinfo **);
|
||||||
int getnameinfo(const struct sockaddr *, socklen_t, char *, size_t, char *, size_t, int);
|
int getnameinfo(const struct sockaddr *, socklen_t, char *, size_t, char *, size_t, int);
|
||||||
int android_getnameinfoforiface(const struct sockaddr *, socklen_t, char *, size_t, char *, size_t, int, const char *);
|
int android_getnameinfoforiface(const struct sockaddr *, socklen_t, char *, size_t, char *, size_t, int, const char *, int);
|
||||||
void freeaddrinfo(struct addrinfo *);
|
void freeaddrinfo(struct addrinfo *);
|
||||||
const char *gai_strerror(int);
|
const char *gai_strerror(int);
|
||||||
void setnetgrent(const char *);
|
void setnetgrent(const char *);
|
||||||
|
@ -1,24 +0,0 @@
|
|||||||
/****************************************************************************
|
|
||||||
****************************************************************************
|
|
||||||
***
|
|
||||||
*** This header was automatically generated from a Linux kernel header
|
|
||||||
*** of the same name, to make information necessary for userspace to
|
|
||||||
*** call into the kernel available to libc. It contains only constants,
|
|
||||||
*** structures, and macros generated from the original header, and thus,
|
|
||||||
*** contains no copyrightable information.
|
|
||||||
***
|
|
||||||
*** To edit the content of this header, modify the corresponding
|
|
||||||
*** source file (e.g. under external/kernel-headers/original/) then
|
|
||||||
*** run bionic/libc/kernel/tools/update_all.py
|
|
||||||
***
|
|
||||||
*** Any manual change here will be lost the next time this script will
|
|
||||||
*** be run. You've been warned!
|
|
||||||
***
|
|
||||||
****************************************************************************
|
|
||||||
****************************************************************************/
|
|
||||||
#ifndef LINUX_KEXEC_H
|
|
||||||
#define LINUX_KEXEC_H
|
|
||||||
struct pt_regs;
|
|
||||||
struct task_struct;
|
|
||||||
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
|
||||||
#endif
|
|
49
libc/kernel/common/uapi/linux/kexec.h
Normal file
49
libc/kernel/common/uapi/linux/kexec.h
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
****************************************************************************
|
||||||
|
***
|
||||||
|
*** This header was automatically generated from a Linux kernel header
|
||||||
|
*** of the same name, to make information necessary for userspace to
|
||||||
|
*** call into the kernel available to libc. It contains only constants,
|
||||||
|
*** structures, and macros generated from the original header, and thus,
|
||||||
|
*** contains no copyrightable information.
|
||||||
|
***
|
||||||
|
*** To edit the content of this header, modify the corresponding
|
||||||
|
*** source file (e.g. under external/kernel-headers/original/) then
|
||||||
|
*** run bionic/libc/kernel/tools/update_all.py
|
||||||
|
***
|
||||||
|
*** Any manual change here will be lost the next time this script will
|
||||||
|
*** be run. You've been warned!
|
||||||
|
***
|
||||||
|
****************************************************************************
|
||||||
|
****************************************************************************/
|
||||||
|
#ifndef _UAPILINUX_KEXEC_H
|
||||||
|
#define _UAPILINUX_KEXEC_H
|
||||||
|
#include <linux/types.h>
|
||||||
|
#define KEXEC_ON_CRASH 0x00000001
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
#define KEXEC_PRESERVE_CONTEXT 0x00000002
|
||||||
|
#define KEXEC_ARCH_MASK 0xffff0000
|
||||||
|
#define KEXEC_ARCH_DEFAULT ( 0 << 16)
|
||||||
|
#define KEXEC_ARCH_386 ( 3 << 16)
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
#define KEXEC_ARCH_X86_64 (62 << 16)
|
||||||
|
#define KEXEC_ARCH_PPC (20 << 16)
|
||||||
|
#define KEXEC_ARCH_PPC64 (21 << 16)
|
||||||
|
#define KEXEC_ARCH_IA_64 (50 << 16)
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
#define KEXEC_ARCH_ARM (40 << 16)
|
||||||
|
#define KEXEC_ARCH_S390 (22 << 16)
|
||||||
|
#define KEXEC_ARCH_SH (42 << 16)
|
||||||
|
#define KEXEC_ARCH_MIPS_LE (10 << 16)
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
#define KEXEC_ARCH_MIPS ( 8 << 16)
|
||||||
|
#define KEXEC_SEGMENT_MAX 16
|
||||||
|
struct kexec_segment {
|
||||||
|
const void *buf;
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
size_t bufsz;
|
||||||
|
const void *mem;
|
||||||
|
size_t memsz;
|
||||||
|
};
|
||||||
|
/* WARNING: DO NOT EDIT, AUTO-GENERATED CODE - SEE TOP FOR INSTRUCTIONS */
|
||||||
|
#endif
|
@ -126,7 +126,7 @@ static struct hostent *_gethtbyname2(const char *, int);
|
|||||||
static int _dns_gethtbyaddr(void *, void *, va_list);
|
static int _dns_gethtbyaddr(void *, void *, va_list);
|
||||||
static int _dns_gethtbyname(void *, void *, va_list);
|
static int _dns_gethtbyname(void *, void *, va_list);
|
||||||
|
|
||||||
static struct hostent *gethostbyname_internal(const char *, int, res_state, const char *);
|
static struct hostent *gethostbyname_internal(const char *, int, res_state, const char *, int);
|
||||||
|
|
||||||
static const ns_src default_dns_files[] = {
|
static const ns_src default_dns_files[] = {
|
||||||
{ NSSRC_FILES, NS_SUCCESS },
|
{ NSSRC_FILES, NS_SUCCESS },
|
||||||
@ -497,13 +497,13 @@ gethostbyname(const char *name)
|
|||||||
|
|
||||||
/* try IPv6 first - if that fails do IPv4 */
|
/* try IPv6 first - if that fails do IPv4 */
|
||||||
if (res->options & RES_USE_INET6) {
|
if (res->options & RES_USE_INET6) {
|
||||||
hp = gethostbyname_internal(name, AF_INET6, res, NULL);
|
hp = gethostbyname_internal(name, AF_INET6, res, NULL, 0);
|
||||||
if (hp) {
|
if (hp) {
|
||||||
__res_put_state(res);
|
__res_put_state(res);
|
||||||
return hp;
|
return hp;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hp = gethostbyname_internal(name, AF_INET, res, NULL);
|
hp = gethostbyname_internal(name, AF_INET, res, NULL, 0);
|
||||||
__res_put_state(res);
|
__res_put_state(res);
|
||||||
return hp;
|
return hp;
|
||||||
}
|
}
|
||||||
@ -511,18 +511,18 @@ gethostbyname(const char *name)
|
|||||||
struct hostent *
|
struct hostent *
|
||||||
gethostbyname2(const char *name, int af)
|
gethostbyname2(const char *name, int af)
|
||||||
{
|
{
|
||||||
return android_gethostbynameforiface(name, af, NULL);
|
return android_gethostbynameforiface(name, af, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct hostent *
|
struct hostent *
|
||||||
android_gethostbynameforiface(const char *name, int af, const char *iface)
|
android_gethostbynameforiface(const char *name, int af, const char *iface, int mark)
|
||||||
{
|
{
|
||||||
struct hostent *hp;
|
struct hostent *hp;
|
||||||
res_state res = __res_get_state();
|
res_state res = __res_get_state();
|
||||||
|
|
||||||
if (res == NULL)
|
if (res == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
hp = gethostbyname_internal(name, af, res, iface);
|
hp = gethostbyname_internal(name, af, res, iface, mark);
|
||||||
__res_put_state(res);
|
__res_put_state(res);
|
||||||
return hp;
|
return hp;
|
||||||
}
|
}
|
||||||
@ -741,7 +741,7 @@ gethostbyname_internal_real(const char *name, int af, res_state res)
|
|||||||
|
|
||||||
// very similar in proxy-ness to android_getaddrinfo_proxy
|
// very similar in proxy-ness to android_getaddrinfo_proxy
|
||||||
static struct hostent *
|
static struct hostent *
|
||||||
gethostbyname_internal(const char *name, int af, res_state res, const char *iface)
|
gethostbyname_internal(const char *name, int af, res_state res, const char *iface, int mark)
|
||||||
{
|
{
|
||||||
const char *cache_mode = getenv("ANDROID_DNS_MODE");
|
const char *cache_mode = getenv("ANDROID_DNS_MODE");
|
||||||
FILE* proxy = NULL;
|
FILE* proxy = NULL;
|
||||||
@ -749,6 +749,7 @@ gethostbyname_internal(const char *name, int af, res_state res, const char *ifac
|
|||||||
|
|
||||||
if (cache_mode != NULL && strcmp(cache_mode, "local") == 0) {
|
if (cache_mode != NULL && strcmp(cache_mode, "local") == 0) {
|
||||||
res_setiface(res, iface);
|
res_setiface(res, iface);
|
||||||
|
res_setmark(res, mark);
|
||||||
return gethostbyname_internal_real(name, af, res);
|
return gethostbyname_internal_real(name, af, res);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -780,7 +781,7 @@ exit:
|
|||||||
|
|
||||||
struct hostent *
|
struct hostent *
|
||||||
android_gethostbyaddrforiface_proxy(const void *addr,
|
android_gethostbyaddrforiface_proxy(const void *addr,
|
||||||
socklen_t len, int af, const char* iface)
|
socklen_t len, int af, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
struct hostent *result = NULL;
|
struct hostent *result = NULL;
|
||||||
FILE* proxy = android_open_proxy();
|
FILE* proxy = android_open_proxy();
|
||||||
@ -810,7 +811,7 @@ exit:
|
|||||||
|
|
||||||
struct hostent *
|
struct hostent *
|
||||||
android_gethostbyaddrforiface_real(const void *addr,
|
android_gethostbyaddrforiface_real(const void *addr,
|
||||||
socklen_t len, int af, const char* iface)
|
socklen_t len, int af, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
const u_char *uaddr = (const u_char *)addr;
|
const u_char *uaddr = (const u_char *)addr;
|
||||||
socklen_t size;
|
socklen_t size;
|
||||||
@ -858,28 +859,28 @@ android_gethostbyaddrforiface_real(const void *addr,
|
|||||||
hp = NULL;
|
hp = NULL;
|
||||||
h_errno = NETDB_INTERNAL;
|
h_errno = NETDB_INTERNAL;
|
||||||
if (nsdispatch(&hp, dtab, NSDB_HOSTS, "gethostbyaddr",
|
if (nsdispatch(&hp, dtab, NSDB_HOSTS, "gethostbyaddr",
|
||||||
default_dns_files, uaddr, len, af, iface) != NS_SUCCESS)
|
default_dns_files, uaddr, len, af, iface, mark) != NS_SUCCESS)
|
||||||
return NULL;
|
return NULL;
|
||||||
h_errno = NETDB_SUCCESS;
|
h_errno = NETDB_SUCCESS;
|
||||||
return hp;
|
return hp;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct hostent *
|
struct hostent *
|
||||||
android_gethostbyaddrforiface(const void *addr, socklen_t len, int af, const char* iface)
|
android_gethostbyaddrforiface(const void *addr, socklen_t len, int af, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
const char *cache_mode = getenv("ANDROID_DNS_MODE");
|
const char *cache_mode = getenv("ANDROID_DNS_MODE");
|
||||||
|
|
||||||
if (cache_mode == NULL || strcmp(cache_mode, "local") != 0) {
|
if (cache_mode == NULL || strcmp(cache_mode, "local") != 0) {
|
||||||
return android_gethostbyaddrforiface_proxy(addr, len, af, iface);
|
return android_gethostbyaddrforiface_proxy(addr, len, af, iface, mark);
|
||||||
} else {
|
} else {
|
||||||
return android_gethostbyaddrforiface_real(addr,len, af,iface);
|
return android_gethostbyaddrforiface_real(addr,len, af, iface, mark);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct hostent *
|
struct hostent *
|
||||||
gethostbyaddr(const void *addr, socklen_t len, int af)
|
gethostbyaddr(const void *addr, socklen_t len, int af)
|
||||||
{
|
{
|
||||||
return android_gethostbyaddrforiface(addr, len, af, NULL);
|
return android_gethostbyaddrforiface(addr, len, af, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1315,6 +1316,7 @@ _dns_gethtbyaddr(void *rv, void *cb_data, va_list ap)
|
|||||||
int len, af, advance;
|
int len, af, advance;
|
||||||
res_state res;
|
res_state res;
|
||||||
const char* iface;
|
const char* iface;
|
||||||
|
int mark;
|
||||||
res_static rs = __res_get_static();
|
res_static rs = __res_get_static();
|
||||||
|
|
||||||
assert(rv != NULL);
|
assert(rv != NULL);
|
||||||
@ -1323,6 +1325,7 @@ _dns_gethtbyaddr(void *rv, void *cb_data, va_list ap)
|
|||||||
len = va_arg(ap, int);
|
len = va_arg(ap, int);
|
||||||
af = va_arg(ap, int);
|
af = va_arg(ap, int);
|
||||||
iface = va_arg(ap, char *);
|
iface = va_arg(ap, char *);
|
||||||
|
mark = va_arg(ap, int);
|
||||||
|
|
||||||
switch (af) {
|
switch (af) {
|
||||||
case AF_INET:
|
case AF_INET:
|
||||||
@ -1365,6 +1368,7 @@ _dns_gethtbyaddr(void *rv, void *cb_data, va_list ap)
|
|||||||
return NS_NOTFOUND;
|
return NS_NOTFOUND;
|
||||||
}
|
}
|
||||||
res_setiface(res, iface);
|
res_setiface(res, iface);
|
||||||
|
res_setmark(res, mark);
|
||||||
n = res_nquery(res, qbuf, C_IN, T_PTR, buf->buf, sizeof(buf->buf));
|
n = res_nquery(res, qbuf, C_IN, T_PTR, buf->buf, sizeof(buf->buf));
|
||||||
if (n < 0) {
|
if (n < 0) {
|
||||||
free(buf);
|
free(buf);
|
||||||
|
@ -215,7 +215,7 @@ struct res_target {
|
|||||||
|
|
||||||
static int str2number(const char *);
|
static int str2number(const char *);
|
||||||
static int explore_fqdn(const struct addrinfo *, const char *,
|
static int explore_fqdn(const struct addrinfo *, const char *,
|
||||||
const char *, struct addrinfo **, const char *iface);
|
const char *, struct addrinfo **, const char *iface, int mark);
|
||||||
static int explore_null(const struct addrinfo *,
|
static int explore_null(const struct addrinfo *,
|
||||||
const char *, struct addrinfo **);
|
const char *, struct addrinfo **);
|
||||||
static int explore_numeric(const struct addrinfo *, const char *,
|
static int explore_numeric(const struct addrinfo *, const char *,
|
||||||
@ -578,12 +578,12 @@ int
|
|||||||
getaddrinfo(const char *hostname, const char *servname,
|
getaddrinfo(const char *hostname, const char *servname,
|
||||||
const struct addrinfo *hints, struct addrinfo **res)
|
const struct addrinfo *hints, struct addrinfo **res)
|
||||||
{
|
{
|
||||||
return android_getaddrinfoforiface(hostname, servname, hints, NULL, res);
|
return android_getaddrinfoforiface(hostname, servname, hints, NULL, 0, res);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
android_getaddrinfoforiface(const char *hostname, const char *servname,
|
android_getaddrinfoforiface(const char *hostname, const char *servname,
|
||||||
const struct addrinfo *hints, const char *iface, struct addrinfo **res)
|
const struct addrinfo *hints, const char *iface, int mark, struct addrinfo **res)
|
||||||
{
|
{
|
||||||
struct addrinfo sentinel;
|
struct addrinfo sentinel;
|
||||||
struct addrinfo *cur;
|
struct addrinfo *cur;
|
||||||
@ -762,7 +762,7 @@ android_getaddrinfoforiface(const char *hostname, const char *servname,
|
|||||||
pai->ai_protocol = ex->e_protocol;
|
pai->ai_protocol = ex->e_protocol;
|
||||||
|
|
||||||
error = explore_fqdn(pai, hostname, servname,
|
error = explore_fqdn(pai, hostname, servname,
|
||||||
&cur->ai_next, iface);
|
&cur->ai_next, iface, mark);
|
||||||
|
|
||||||
while (cur && cur->ai_next)
|
while (cur && cur->ai_next)
|
||||||
cur = cur->ai_next;
|
cur = cur->ai_next;
|
||||||
@ -795,7 +795,7 @@ android_getaddrinfoforiface(const char *hostname, const char *servname,
|
|||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
explore_fqdn(const struct addrinfo *pai, const char *hostname,
|
explore_fqdn(const struct addrinfo *pai, const char *hostname,
|
||||||
const char *servname, struct addrinfo **res, const char *iface)
|
const char *servname, struct addrinfo **res, const char *iface, int mark)
|
||||||
{
|
{
|
||||||
struct addrinfo *result;
|
struct addrinfo *result;
|
||||||
struct addrinfo *cur;
|
struct addrinfo *cur;
|
||||||
@ -821,7 +821,7 @@ explore_fqdn(const struct addrinfo *pai, const char *hostname,
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
switch (nsdispatch(&result, dtab, NSDB_HOSTS, "getaddrinfo",
|
switch (nsdispatch(&result, dtab, NSDB_HOSTS, "getaddrinfo",
|
||||||
default_dns_files, hostname, pai, iface)) {
|
default_dns_files, hostname, pai, iface, mark)) {
|
||||||
case NS_TRYAGAIN:
|
case NS_TRYAGAIN:
|
||||||
error = EAI_AGAIN;
|
error = EAI_AGAIN;
|
||||||
goto free;
|
goto free;
|
||||||
@ -1892,10 +1892,12 @@ _dns_getaddrinfo(void *rv, void *cb_data, va_list ap)
|
|||||||
struct res_target q, q2;
|
struct res_target q, q2;
|
||||||
res_state res;
|
res_state res;
|
||||||
const char* iface;
|
const char* iface;
|
||||||
|
int mark;
|
||||||
|
|
||||||
name = va_arg(ap, char *);
|
name = va_arg(ap, char *);
|
||||||
pai = va_arg(ap, const struct addrinfo *);
|
pai = va_arg(ap, const struct addrinfo *);
|
||||||
iface = va_arg(ap, char *);
|
iface = va_arg(ap, char *);
|
||||||
|
mark = va_arg(ap, int);
|
||||||
//fprintf(stderr, "_dns_getaddrinfo() name = '%s'\n", name);
|
//fprintf(stderr, "_dns_getaddrinfo() name = '%s'\n", name);
|
||||||
|
|
||||||
memset(&q, 0, sizeof(q));
|
memset(&q, 0, sizeof(q));
|
||||||
@ -1983,6 +1985,7 @@ _dns_getaddrinfo(void *rv, void *cb_data, va_list ap)
|
|||||||
* and have a cache hit that would be wasted, so we do the rest there on miss
|
* and have a cache hit that would be wasted, so we do the rest there on miss
|
||||||
*/
|
*/
|
||||||
res_setiface(res, iface);
|
res_setiface(res, iface);
|
||||||
|
res_setmark(res, mark);
|
||||||
if (res_searchN(name, &q, res) < 0) {
|
if (res_searchN(name, &q, res) < 0) {
|
||||||
__res_put_state(res);
|
__res_put_state(res);
|
||||||
free(buf);
|
free(buf);
|
||||||
|
@ -93,7 +93,7 @@ struct sockinet {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static int getnameinfo_inet(const struct sockaddr *, socklen_t, char *,
|
static int getnameinfo_inet(const struct sockaddr *, socklen_t, char *,
|
||||||
socklen_t, char *, socklen_t, int, const char*);
|
socklen_t, char *, socklen_t, int, const char*, int);
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
static int ip6_parsenumeric(const struct sockaddr *, const char *, char *,
|
static int ip6_parsenumeric(const struct sockaddr *, const char *, char *,
|
||||||
socklen_t, int);
|
socklen_t, int);
|
||||||
@ -108,16 +108,16 @@ static int getnameinfo_local(const struct sockaddr *, socklen_t, char *,
|
|||||||
*/
|
*/
|
||||||
int getnameinfo(const struct sockaddr* sa, socklen_t salen, char* host, size_t hostlen, char* serv, size_t servlen, int flags)
|
int getnameinfo(const struct sockaddr* sa, socklen_t salen, char* host, size_t hostlen, char* serv, size_t servlen, int flags)
|
||||||
{
|
{
|
||||||
return android_getnameinfoforiface(sa, salen, host, hostlen, serv, servlen, flags, NULL);
|
return android_getnameinfoforiface(sa, salen, host, hostlen, serv, servlen, flags, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
int android_getnameinfoforiface(const struct sockaddr* sa, socklen_t salen, char* host, size_t hostlen, char* serv, size_t servlen, int flags, const char* iface)
|
int android_getnameinfoforiface(const struct sockaddr* sa, socklen_t salen, char* host, size_t hostlen, char* serv, size_t servlen, int flags, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
switch (sa->sa_family) {
|
switch (sa->sa_family) {
|
||||||
case AF_INET:
|
case AF_INET:
|
||||||
case AF_INET6:
|
case AF_INET6:
|
||||||
return getnameinfo_inet(sa, salen, host, hostlen,
|
return getnameinfo_inet(sa, salen, host, hostlen,
|
||||||
serv, servlen, flags, iface);
|
serv, servlen, flags, iface, mark);
|
||||||
case AF_LOCAL:
|
case AF_LOCAL:
|
||||||
return getnameinfo_local(sa, salen, host, hostlen,
|
return getnameinfo_local(sa, salen, host, hostlen,
|
||||||
serv, servlen, flags);
|
serv, servlen, flags);
|
||||||
@ -158,10 +158,10 @@ getnameinfo_local(const struct sockaddr *sa, socklen_t salen,
|
|||||||
* the address. On failure -1 is returned in which case
|
* the address. On failure -1 is returned in which case
|
||||||
* normal execution flow shall continue. */
|
* normal execution flow shall continue. */
|
||||||
static int
|
static int
|
||||||
android_gethostbyaddr_proxy(char* nameBuf, size_t nameBufLen, const void *addr, socklen_t addrLen, int addrFamily, const char* iface)
|
android_gethostbyaddr_proxy(char* nameBuf, size_t nameBufLen, const void *addr, socklen_t addrLen, int addrFamily, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
struct hostent *hostResult =
|
struct hostent *hostResult =
|
||||||
android_gethostbyaddrforiface_proxy(addr, addrLen, addrFamily, iface);
|
android_gethostbyaddrforiface_proxy(addr, addrLen, addrFamily, iface, mark);
|
||||||
|
|
||||||
if (hostResult == NULL) return 0;
|
if (hostResult == NULL) return 0;
|
||||||
|
|
||||||
@ -179,7 +179,7 @@ static int
|
|||||||
getnameinfo_inet(const struct sockaddr* sa, socklen_t salen,
|
getnameinfo_inet(const struct sockaddr* sa, socklen_t salen,
|
||||||
char *host, socklen_t hostlen,
|
char *host, socklen_t hostlen,
|
||||||
char *serv, socklen_t servlen,
|
char *serv, socklen_t servlen,
|
||||||
int flags, const char* iface)
|
int flags, const char* iface, int mark)
|
||||||
{
|
{
|
||||||
const struct afd *afd;
|
const struct afd *afd;
|
||||||
struct servent *sp;
|
struct servent *sp;
|
||||||
@ -321,14 +321,15 @@ getnameinfo_inet(const struct sockaddr* sa, socklen_t salen,
|
|||||||
char android_proxy_buf[MAXDNAME];
|
char android_proxy_buf[MAXDNAME];
|
||||||
|
|
||||||
int hostnamelen = android_gethostbyaddr_proxy(android_proxy_buf,
|
int hostnamelen = android_gethostbyaddr_proxy(android_proxy_buf,
|
||||||
MAXDNAME, addr, afd->a_addrlen, afd->a_af, iface);
|
MAXDNAME, addr, afd->a_addrlen, afd->a_af, iface, mark);
|
||||||
if (hostnamelen > 0) {
|
if (hostnamelen > 0) {
|
||||||
hp = &android_proxy_hostent;
|
hp = &android_proxy_hostent;
|
||||||
hp->h_name = android_proxy_buf;
|
hp->h_name = android_proxy_buf;
|
||||||
} else if (!hostnamelen) {
|
} else if (!hostnamelen) {
|
||||||
hp = NULL;
|
hp = NULL;
|
||||||
} else {
|
} else {
|
||||||
hp = android_gethostbyaddrforiface(addr, afd->a_addrlen, afd->a_af, iface);
|
hp = android_gethostbyaddrforiface(addr, afd->a_addrlen, afd->a_af,
|
||||||
|
iface, mark);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hp) {
|
if (hp) {
|
||||||
|
@ -1258,6 +1258,12 @@ typedef struct resolv_pidiface_info {
|
|||||||
char ifname[IF_NAMESIZE + 1];
|
char ifname[IF_NAMESIZE + 1];
|
||||||
struct resolv_pidiface_info* next;
|
struct resolv_pidiface_info* next;
|
||||||
} PidIfaceInfo;
|
} PidIfaceInfo;
|
||||||
|
typedef struct resolv_uidiface_info {
|
||||||
|
int uid_start;
|
||||||
|
int uid_end;
|
||||||
|
char ifname[IF_NAMESIZE + 1];
|
||||||
|
struct resolv_uidiface_info* next;
|
||||||
|
} UidIfaceInfo;
|
||||||
|
|
||||||
#define HTABLE_VALID(x) ((x) != NULL && (x) != HTABLE_DELETED)
|
#define HTABLE_VALID(x) ((x) != NULL && (x) != HTABLE_DELETED)
|
||||||
|
|
||||||
@ -1796,6 +1802,9 @@ static struct resolv_cache_info _res_cache_list;
|
|||||||
// List of pid iface pairs
|
// List of pid iface pairs
|
||||||
static struct resolv_pidiface_info _res_pidiface_list;
|
static struct resolv_pidiface_info _res_pidiface_list;
|
||||||
|
|
||||||
|
// List of uid iface pairs
|
||||||
|
static struct resolv_uidiface_info _res_uidiface_list;
|
||||||
|
|
||||||
// name of the current default inteface
|
// name of the current default inteface
|
||||||
static char _res_default_ifname[IF_NAMESIZE + 1];
|
static char _res_default_ifname[IF_NAMESIZE + 1];
|
||||||
|
|
||||||
@ -1805,6 +1814,9 @@ static pthread_mutex_t _res_cache_list_lock;
|
|||||||
// lock protecting the _res_pid_iface_list
|
// lock protecting the _res_pid_iface_list
|
||||||
static pthread_mutex_t _res_pidiface_list_lock;
|
static pthread_mutex_t _res_pidiface_list_lock;
|
||||||
|
|
||||||
|
// lock protecting the _res_uidiface_list
|
||||||
|
static pthread_mutex_t _res_uidiface_list_lock;
|
||||||
|
|
||||||
/* lookup the default interface name */
|
/* lookup the default interface name */
|
||||||
static char *_get_default_iface_locked();
|
static char *_get_default_iface_locked();
|
||||||
/* find the first cache that has an associated interface and return the name of the interface */
|
/* find the first cache that has an associated interface and return the name of the interface */
|
||||||
@ -1833,12 +1845,19 @@ static struct in_addr* _get_addr_locked(const char * ifname);
|
|||||||
/* return 1 if the provided list of name servers differs from the list of name servers
|
/* return 1 if the provided list of name servers differs from the list of name servers
|
||||||
* currently attached to the provided cache_info */
|
* currently attached to the provided cache_info */
|
||||||
static int _resolv_is_nameservers_equal_locked(struct resolv_cache_info* cache_info,
|
static int _resolv_is_nameservers_equal_locked(struct resolv_cache_info* cache_info,
|
||||||
char** servers, int numservers);
|
const char** servers, int numservers);
|
||||||
/* remove a resolv_pidiface_info structure from _res_pidiface_list */
|
/* remove a resolv_pidiface_info structure from _res_pidiface_list */
|
||||||
static void _remove_pidiface_info_locked(int pid);
|
static void _remove_pidiface_info_locked(int pid);
|
||||||
/* get a resolv_pidiface_info structure from _res_pidiface_list with a certain pid */
|
/* get a resolv_pidiface_info structure from _res_pidiface_list with a certain pid */
|
||||||
static struct resolv_pidiface_info* _get_pid_iface_info_locked(int pid);
|
static struct resolv_pidiface_info* _get_pid_iface_info_locked(int pid);
|
||||||
|
|
||||||
|
/* remove a resolv_pidiface_info structure from _res_uidiface_list */
|
||||||
|
static int _remove_uidiface_info_locked(int uid_start, int uid_end);
|
||||||
|
/* check if a range [low,high] overlaps with any already existing ranges in the uid=>iface map*/
|
||||||
|
static int _resolv_check_uid_range_overlap_locked(int uid_start, int uid_end);
|
||||||
|
/* get a resolv_uidiface_info structure from _res_uidiface_list with a certain uid */
|
||||||
|
static struct resolv_uidiface_info* _get_uid_iface_info_locked(int uid);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
_res_cache_init(void)
|
_res_cache_init(void)
|
||||||
{
|
{
|
||||||
@ -1852,8 +1871,10 @@ _res_cache_init(void)
|
|||||||
memset(&_res_default_ifname, 0, sizeof(_res_default_ifname));
|
memset(&_res_default_ifname, 0, sizeof(_res_default_ifname));
|
||||||
memset(&_res_cache_list, 0, sizeof(_res_cache_list));
|
memset(&_res_cache_list, 0, sizeof(_res_cache_list));
|
||||||
memset(&_res_pidiface_list, 0, sizeof(_res_pidiface_list));
|
memset(&_res_pidiface_list, 0, sizeof(_res_pidiface_list));
|
||||||
|
memset(&_res_uidiface_list, 0, sizeof(_res_uidiface_list));
|
||||||
pthread_mutex_init(&_res_cache_list_lock, NULL);
|
pthread_mutex_init(&_res_cache_list_lock, NULL);
|
||||||
pthread_mutex_init(&_res_pidiface_list_lock, NULL);
|
pthread_mutex_init(&_res_pidiface_list_lock, NULL);
|
||||||
|
pthread_mutex_init(&_res_uidiface_list_lock, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct resolv_cache*
|
struct resolv_cache*
|
||||||
@ -2076,7 +2097,7 @@ _resolv_set_default_iface(const char* ifname)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
_resolv_set_nameservers_for_iface(const char* ifname, char** servers, int numservers,
|
_resolv_set_nameservers_for_iface(const char* ifname, const char** servers, int numservers,
|
||||||
const char *domains)
|
const char *domains)
|
||||||
{
|
{
|
||||||
int i, rt, index;
|
int i, rt, index;
|
||||||
@ -2149,7 +2170,7 @@ _resolv_set_nameservers_for_iface(const char* ifname, char** servers, int numser
|
|||||||
|
|
||||||
static int
|
static int
|
||||||
_resolv_is_nameservers_equal_locked(struct resolv_cache_info* cache_info,
|
_resolv_is_nameservers_equal_locked(struct resolv_cache_info* cache_info,
|
||||||
char** servers, int numservers)
|
const char** servers, int numservers)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
char** ns;
|
char** ns;
|
||||||
@ -2271,8 +2292,8 @@ _resolv_set_addr_of_iface(const char* ifname, struct in_addr* addr)
|
|||||||
memcpy(&cache_info->ifaddr, addr, sizeof(*addr));
|
memcpy(&cache_info->ifaddr, addr, sizeof(*addr));
|
||||||
|
|
||||||
if (DEBUG) {
|
if (DEBUG) {
|
||||||
char* addr_s = inet_ntoa(cache_info->ifaddr);
|
XLOG("address of interface %s is %s\n",
|
||||||
XLOG("address of interface %s is %s\n", ifname, addr_s);
|
ifname, inet_ntoa(cache_info->ifaddr));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pthread_mutex_unlock(&_res_cache_list_lock);
|
pthread_mutex_unlock(&_res_cache_list_lock);
|
||||||
@ -2411,20 +2432,177 @@ _resolv_get_pids_associated_interface(int pid, char* buff, int buffLen)
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
static int
|
||||||
_resolv_get_default_iface(char* buff, int buffLen)
|
_remove_uidiface_info_locked(int uid_start, int uid_end) {
|
||||||
|
struct resolv_uidiface_info* result = _res_uidiface_list.next;
|
||||||
|
struct resolv_uidiface_info* prev = &_res_uidiface_list;
|
||||||
|
|
||||||
|
while (result != NULL && result->uid_start != uid_start && result->uid_end != uid_end) {
|
||||||
|
prev = result;
|
||||||
|
result = result->next;
|
||||||
|
}
|
||||||
|
if (prev != NULL && result != NULL) {
|
||||||
|
prev->next = result->next;
|
||||||
|
free(result);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
errno = EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct resolv_uidiface_info*
|
||||||
|
_get_uid_iface_info_locked(int uid)
|
||||||
|
{
|
||||||
|
struct resolv_uidiface_info* result = _res_uidiface_list.next;
|
||||||
|
while (result != NULL && !(result->uid_start <= uid && result->uid_end >= uid)) {
|
||||||
|
result = result->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
_resolv_check_uid_range_overlap_locked(int uid_start, int uid_end)
|
||||||
|
{
|
||||||
|
struct resolv_uidiface_info* cur = _res_uidiface_list.next;
|
||||||
|
while (cur != NULL) {
|
||||||
|
if (cur->uid_start <= uid_end && cur->uid_end >= uid_start) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
cur = cur->next;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_resolv_clear_iface_uid_range_mapping()
|
||||||
|
{
|
||||||
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
|
pthread_mutex_lock(&_res_uidiface_list_lock);
|
||||||
|
struct resolv_uidiface_info *current = _res_uidiface_list.next;
|
||||||
|
struct resolv_uidiface_info *next;
|
||||||
|
while (current != NULL) {
|
||||||
|
next = current->next;
|
||||||
|
free(current);
|
||||||
|
current = next;
|
||||||
|
}
|
||||||
|
_res_uidiface_list.next = NULL;
|
||||||
|
pthread_mutex_unlock(&_res_uidiface_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_resolv_clear_iface_pid_mapping()
|
||||||
|
{
|
||||||
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
|
pthread_mutex_lock(&_res_pidiface_list_lock);
|
||||||
|
struct resolv_pidiface_info *current = _res_pidiface_list.next;
|
||||||
|
struct resolv_pidiface_info *next;
|
||||||
|
while (current != NULL) {
|
||||||
|
next = current->next;
|
||||||
|
free(current);
|
||||||
|
current = next;
|
||||||
|
}
|
||||||
|
_res_pidiface_list.next = NULL;
|
||||||
|
pthread_mutex_unlock(&_res_pidiface_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_resolv_set_iface_for_uid_range(const char* ifname, int uid_start, int uid_end)
|
||||||
|
{
|
||||||
|
int rv = 0;
|
||||||
|
struct resolv_uidiface_info* uidiface_info;
|
||||||
|
// make sure the uid iface list is created
|
||||||
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
|
if (uid_start > uid_end) {
|
||||||
|
errno = EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
pthread_mutex_lock(&_res_uidiface_list_lock);
|
||||||
|
//check that we aren't adding an overlapping range
|
||||||
|
if (!_resolv_check_uid_range_overlap_locked(uid_start, uid_end)) {
|
||||||
|
uidiface_info = calloc(sizeof(*uidiface_info), 1);
|
||||||
|
if (uidiface_info) {
|
||||||
|
uidiface_info->uid_start = uid_start;
|
||||||
|
uidiface_info->uid_end = uid_end;
|
||||||
|
int len = sizeof(uidiface_info->ifname);
|
||||||
|
strncpy(uidiface_info->ifname, ifname, len - 1);
|
||||||
|
uidiface_info->ifname[len - 1] = '\0';
|
||||||
|
|
||||||
|
uidiface_info->next = _res_uidiface_list.next;
|
||||||
|
_res_uidiface_list.next = uidiface_info;
|
||||||
|
|
||||||
|
XLOG("_resolv_set_iface_for_uid_range: [%d,%d], iface %s\n", uid_start, uid_end,
|
||||||
|
ifname);
|
||||||
|
} else {
|
||||||
|
XLOG("_resolv_set_iface_for_uid_range failing calloc\n");
|
||||||
|
rv = -1;
|
||||||
|
errno = EINVAL;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
XLOG("_resolv_set_iface_for_uid_range range [%d,%d] overlaps\n", uid_start, uid_end);
|
||||||
|
rv = -1;
|
||||||
|
errno = EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
pthread_mutex_unlock(&_res_uidiface_list_lock);
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_resolv_clear_iface_for_uid_range(int uid_start, int uid_end)
|
||||||
|
{
|
||||||
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
|
pthread_mutex_lock(&_res_uidiface_list_lock);
|
||||||
|
|
||||||
|
int rv = _remove_uidiface_info_locked(uid_start, uid_end);
|
||||||
|
|
||||||
|
XLOG("_resolv_clear_iface_for_uid_range: [%d,%d]\n", uid_start, uid_end);
|
||||||
|
|
||||||
|
pthread_mutex_unlock(&_res_uidiface_list_lock);
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_resolv_get_uids_associated_interface(int uid, char* buff, int buffLen)
|
||||||
{
|
{
|
||||||
char* ifname;
|
|
||||||
int len = 0;
|
int len = 0;
|
||||||
|
|
||||||
if (!buff || buffLen == 0) {
|
if (!buff) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
|
pthread_mutex_lock(&_res_uidiface_list_lock);
|
||||||
|
|
||||||
|
struct resolv_uidiface_info* uidiface_info = _get_uid_iface_info_locked(uid);
|
||||||
|
buff[0] = '\0';
|
||||||
|
if (uidiface_info) {
|
||||||
|
len = strlen(uidiface_info->ifname);
|
||||||
|
if (len < buffLen) {
|
||||||
|
strncpy(buff, uidiface_info->ifname, len);
|
||||||
|
buff[len] = '\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
XLOG("_resolv_get_uids_associated_interface buff: %s\n", buff);
|
||||||
|
|
||||||
|
pthread_mutex_unlock(&_res_uidiface_list_lock);
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t
|
||||||
|
_resolv_get_default_iface(char* buff, size_t buffLen)
|
||||||
|
{
|
||||||
|
if (!buff || buffLen == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
pthread_once(&_res_cache_once, _res_cache_init);
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
pthread_mutex_lock(&_res_cache_list_lock);
|
pthread_mutex_lock(&_res_cache_list_lock);
|
||||||
|
|
||||||
ifname = _get_default_iface_locked(); // never null, but may be empty
|
char* ifname = _get_default_iface_locked(); // never null, but may be empty
|
||||||
|
|
||||||
// if default interface not set give up.
|
// if default interface not set give up.
|
||||||
if (ifname[0] == '\0') {
|
if (ifname[0] == '\0') {
|
||||||
@ -2432,7 +2610,7 @@ _resolv_get_default_iface(char* buff, int buffLen)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
len = strlen(ifname);
|
size_t len = strlen(ifname);
|
||||||
if (len < buffLen) {
|
if (len < buffLen) {
|
||||||
strncpy(buff, ifname, len);
|
strncpy(buff, ifname, len);
|
||||||
buff[len] = '\0';
|
buff[len] = '\0';
|
||||||
@ -2445,28 +2623,32 @@ _resolv_get_default_iface(char* buff, int buffLen)
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
void
|
||||||
_resolv_populate_res_for_iface(res_state statp)
|
_resolv_populate_res_for_iface(res_state statp)
|
||||||
{
|
{
|
||||||
int nserv;
|
if (statp == NULL) {
|
||||||
struct resolv_cache_info* info = NULL;
|
return;
|
||||||
|
}
|
||||||
if (statp) {
|
|
||||||
struct addrinfo* ai;
|
|
||||||
|
|
||||||
if (statp->iface[0] == '\0') { // no interface set assign default
|
if (statp->iface[0] == '\0') { // no interface set assign default
|
||||||
_resolv_get_default_iface(statp->iface, sizeof(statp->iface));
|
size_t if_len = _resolv_get_default_iface(statp->iface, sizeof(statp->iface));
|
||||||
|
if (if_len + 1 > sizeof(statp->iface)) {
|
||||||
|
XLOG("%s: INTERNAL_ERROR: can't fit interface name into statp->iface.\n", __FUNCTION__);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (if_len == 0) {
|
||||||
|
XLOG("%s: INTERNAL_ERROR: can't find any suitable interfaces.\n", __FUNCTION__);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pthread_once(&_res_cache_once, _res_cache_init);
|
pthread_once(&_res_cache_once, _res_cache_init);
|
||||||
pthread_mutex_lock(&_res_cache_list_lock);
|
pthread_mutex_lock(&_res_cache_list_lock);
|
||||||
info = _find_cache_info_locked(statp->iface);
|
|
||||||
|
|
||||||
if (info == NULL) {
|
|
||||||
pthread_mutex_unlock(&_res_cache_list_lock);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
struct resolv_cache_info* info = _find_cache_info_locked(statp->iface);
|
||||||
|
if (info != NULL) {
|
||||||
|
int nserv;
|
||||||
|
struct addrinfo* ai;
|
||||||
XLOG("_resolv_populate_res_for_iface: %s\n", statp->iface);
|
XLOG("_resolv_populate_res_for_iface: %s\n", statp->iface);
|
||||||
for (nserv = 0; nserv < MAXNS; nserv++) {
|
for (nserv = 0; nserv < MAXNS; nserv++) {
|
||||||
ai = info->nsaddrinfo[nserv];
|
ai = info->nsaddrinfo[nserv];
|
||||||
@ -2500,8 +2682,6 @@ _resolv_populate_res_for_iface(res_state statp)
|
|||||||
while (pp < statp->dnsrch + MAXDNSRCH && *p != -1) {
|
while (pp < statp->dnsrch + MAXDNSRCH && *p != -1) {
|
||||||
*pp++ = &statp->defdname + *p++;
|
*pp++ = &statp->defdname + *p++;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
pthread_mutex_unlock(&_res_cache_list_lock);
|
pthread_mutex_unlock(&_res_cache_list_lock);
|
||||||
}
|
}
|
||||||
return nserv;
|
|
||||||
}
|
|
||||||
|
@ -806,4 +806,11 @@ void res_setiface(res_state statp, const char* iface)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void res_setmark(res_state statp, int mark)
|
||||||
|
{
|
||||||
|
if (statp != NULL) {
|
||||||
|
statp->_mark = mark;
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif /* ANDROID_CHANGES */
|
#endif /* ANDROID_CHANGES */
|
||||||
|
@ -762,10 +762,13 @@ send_vc(res_state statp,
|
|||||||
if (statp->_vcsock >= 0 && (statp->_flags & RES_F_VC) != 0) {
|
if (statp->_vcsock >= 0 && (statp->_flags & RES_F_VC) != 0) {
|
||||||
struct sockaddr_storage peer;
|
struct sockaddr_storage peer;
|
||||||
socklen_t size = sizeof peer;
|
socklen_t size = sizeof peer;
|
||||||
|
int old_mark;
|
||||||
|
int mark_size = sizeof(old_mark);
|
||||||
if (getpeername(statp->_vcsock,
|
if (getpeername(statp->_vcsock,
|
||||||
(struct sockaddr *)(void *)&peer, &size) < 0 ||
|
(struct sockaddr *)(void *)&peer, &size) < 0 ||
|
||||||
!sock_eq((struct sockaddr *)(void *)&peer, nsap)) {
|
!sock_eq((struct sockaddr *)(void *)&peer, nsap) ||
|
||||||
|
getsockopt(statp->_vcsock, SOL_SOCKET, SO_MARK, &old_mark, &mark_size) < 0 ||
|
||||||
|
old_mark != statp->_mark) {
|
||||||
res_nclose(statp);
|
res_nclose(statp);
|
||||||
statp->_flags &= ~RES_F_VC;
|
statp->_flags &= ~RES_F_VC;
|
||||||
}
|
}
|
||||||
@ -795,6 +798,14 @@ send_vc(res_state statp,
|
|||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (statp->_mark != 0) {
|
||||||
|
if (setsockopt(statp->_vcsock, SOL_SOCKET,
|
||||||
|
SO_MARK, &statp->_mark, sizeof(statp->_mark)) < 0) {
|
||||||
|
*terrno = errno;
|
||||||
|
Perror(statp, stderr, "setsockopt", errno);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
errno = 0;
|
errno = 0;
|
||||||
if (random_bind(statp->_vcsock,nsap->sa_family) < 0) {
|
if (random_bind(statp->_vcsock,nsap->sa_family) < 0) {
|
||||||
*terrno = errno;
|
*terrno = errno;
|
||||||
@ -1070,6 +1081,14 @@ send_dg(res_state statp,
|
|||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (statp->_mark != 0) {
|
||||||
|
if (setsockopt(EXT(statp).nssocks[ns], SOL_SOCKET,
|
||||||
|
SO_MARK, &(statp->_mark), sizeof(statp->_mark)) < 0) {
|
||||||
|
res_nclose(statp);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
#ifndef CANNOT_CONNECT_DGRAM
|
#ifndef CANNOT_CONNECT_DGRAM
|
||||||
/*
|
/*
|
||||||
* On a 4.3BSD+ machine (client and server,
|
* On a 4.3BSD+ machine (client and server,
|
||||||
@ -1097,6 +1116,7 @@ send_dg(res_state statp,
|
|||||||
#endif /* !CANNOT_CONNECT_DGRAM */
|
#endif /* !CANNOT_CONNECT_DGRAM */
|
||||||
Dprint(statp->options & RES_DEBUG,
|
Dprint(statp->options & RES_DEBUG,
|
||||||
(stdout, ";; new DG socket\n"))
|
(stdout, ";; new DG socket\n"))
|
||||||
|
|
||||||
}
|
}
|
||||||
s = EXT(statp).nssocks[ns];
|
s = EXT(statp).nssocks[ns];
|
||||||
#ifndef CANNOT_CONNECT_DGRAM
|
#ifndef CANNOT_CONNECT_DGRAM
|
||||||
|
40
libc/private/bionic_name_mem.h
Normal file
40
libc/private/bionic_name_mem.h
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2013 The Android Open Source Project
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE 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 _BIONIC_NAME_MEM_H
|
||||||
|
#define _BIONIC_NAME_MEM_H
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
|
__BEGIN_DECLS
|
||||||
|
|
||||||
|
int __bionic_name_mem(void *addr, size_t len, const char *name);
|
||||||
|
|
||||||
|
__END_DECLS
|
||||||
|
|
||||||
|
#endif
|
@ -28,6 +28,7 @@
|
|||||||
#ifndef _RESOLV_CACHE_H_
|
#ifndef _RESOLV_CACHE_H_
|
||||||
#define _RESOLV_CACHE_H_
|
#define _RESOLV_CACHE_H_
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
#include <sys/cdefs.h>
|
#include <sys/cdefs.h>
|
||||||
|
|
||||||
struct __res_state;
|
struct __res_state;
|
||||||
@ -77,16 +78,17 @@ extern struct in_addr* _resolv_get_addr_of_default_iface();
|
|||||||
__LIBC_HIDDEN__
|
__LIBC_HIDDEN__
|
||||||
extern struct in_addr* _resolv_get_addr_of_iface(const char* ifname);
|
extern struct in_addr* _resolv_get_addr_of_iface(const char* ifname);
|
||||||
|
|
||||||
/* Copy the name of the default interface to provided buffer.
|
/* Copy the name of the default interface to the provided buffer.
|
||||||
* Return length of buffer on success on failure -1 is returned */
|
* Returns the string length of the default interface,
|
||||||
|
* be that less or more than the buffLen, or 0 if nothing had been written */
|
||||||
__LIBC_HIDDEN__
|
__LIBC_HIDDEN__
|
||||||
extern int _resolv_get_default_iface(char* buff, int buffLen);
|
extern size_t _resolv_get_default_iface(char* buff, size_t buffLen);
|
||||||
|
|
||||||
/* sets the name server addresses to the provided res_state structure. The
|
/* sets the name server addresses to the provided res_state structure. The
|
||||||
* name servers are retrieved from the cache which is associated
|
* name servers are retrieved from the cache which is associated
|
||||||
* with the interface to which the res_state structure is associated */
|
* with the interface to which the res_state structure is associated */
|
||||||
__LIBC_HIDDEN__
|
__LIBC_HIDDEN__
|
||||||
extern int _resolv_populate_res_for_iface(struct __res_state* statp);
|
extern void _resolv_populate_res_for_iface(struct __res_state* statp);
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
RESOLV_CACHE_UNSUPPORTED, /* the cache can't handle that kind of queries */
|
RESOLV_CACHE_UNSUPPORTED, /* the cache can't handle that kind of queries */
|
||||||
|
@ -48,7 +48,7 @@ __BEGIN_DECLS
|
|||||||
extern void _resolv_set_default_iface(const char* ifname);
|
extern void _resolv_set_default_iface(const char* ifname);
|
||||||
|
|
||||||
/* set name servers for an interface */
|
/* set name servers for an interface */
|
||||||
extern void _resolv_set_nameservers_for_iface(const char* ifname, char** servers, int numservers,
|
extern void _resolv_set_nameservers_for_iface(const char* ifname, const char** servers, int numservers,
|
||||||
const char *domains);
|
const char *domains);
|
||||||
|
|
||||||
/* tell resolver of the address of an interface */
|
/* tell resolver of the address of an interface */
|
||||||
@ -66,6 +66,9 @@ extern void _resolv_set_iface_for_pid(const char* ifname, int pid);
|
|||||||
/* clear pid from being associated with an interface */
|
/* clear pid from being associated with an interface */
|
||||||
extern void _resolv_clear_iface_for_pid(int pid);
|
extern void _resolv_clear_iface_for_pid(int pid);
|
||||||
|
|
||||||
|
/* clear the entire mapping of pids to interfaces. */
|
||||||
|
extern void _resolv_clear_iface_pid_mapping();
|
||||||
|
|
||||||
/** Gets the name of the interface to which the pid is attached.
|
/** Gets the name of the interface to which the pid is attached.
|
||||||
* On error, -1 is returned.
|
* On error, -1 is returned.
|
||||||
* If no interface is found, 0 is returned and buff is set to empty ('\0').
|
* If no interface is found, 0 is returned and buff is set to empty ('\0').
|
||||||
@ -75,6 +78,27 @@ extern void _resolv_clear_iface_for_pid(int pid);
|
|||||||
* buffLen Length of buff. An interface is at most IF_NAMESIZE in length */
|
* buffLen Length of buff. An interface is at most IF_NAMESIZE in length */
|
||||||
extern int _resolv_get_pids_associated_interface(int pid, char* buff, int buffLen);
|
extern int _resolv_get_pids_associated_interface(int pid, char* buff, int buffLen);
|
||||||
|
|
||||||
|
|
||||||
|
/** set a uid range to use the name servers of the specified interface
|
||||||
|
* If [low,high] overlaps with an already existing rule -1 is returned */
|
||||||
|
extern int _resolv_set_iface_for_uid_range(const char* ifname, int uid_start, int uid_end);
|
||||||
|
|
||||||
|
/* clear a uid range from being associated with an interface
|
||||||
|
* If the range given is not mapped -1 is returned. */
|
||||||
|
extern int _resolv_clear_iface_for_uid_range(int uid_start, int uid_end);
|
||||||
|
|
||||||
|
/* clear the entire mapping of uid ranges to interfaces. */
|
||||||
|
extern void _resolv_clear_iface_uid_range_mapping();
|
||||||
|
|
||||||
|
/** Gets the name of the interface to which the uid is attached.
|
||||||
|
* On error, -1 is returned.
|
||||||
|
* If no interface is found, 0 is returned and buff is set to empty ('\0').
|
||||||
|
* If an interface is found, the name is copied to buff and the length of the name is returned.
|
||||||
|
* Arguments: uid The uid to find an interface for
|
||||||
|
* buff A buffer to copy the result to
|
||||||
|
* buffLen Length of buff. An interface is at most IF_NAMESIZE in length */
|
||||||
|
extern int _resolv_get_uids_associated_interface(int uid, char* buff, int buffLen);
|
||||||
|
|
||||||
#endif /* _BIONIC_RESOLV_IFACE_FUNCTIONS_DECLARED */
|
#endif /* _BIONIC_RESOLV_IFACE_FUNCTIONS_DECLARED */
|
||||||
|
|
||||||
__END_DECLS
|
__END_DECLS
|
||||||
|
@ -175,6 +175,7 @@ struct __res_state {
|
|||||||
res_send_qhook qhook; /* query hook */
|
res_send_qhook qhook; /* query hook */
|
||||||
res_send_rhook rhook; /* response hook */
|
res_send_rhook rhook; /* response hook */
|
||||||
int res_h_errno; /* last one set for this context */
|
int res_h_errno; /* last one set for this context */
|
||||||
|
int _mark; /* If non-0 SET_MARK to _mark on all request sockets */
|
||||||
int _vcsock; /* PRIVATE: for res_send VC i/o */
|
int _vcsock; /* PRIVATE: for res_send VC i/o */
|
||||||
u_int _flags; /* PRIVATE: see below */
|
u_int _flags; /* PRIVATE: see below */
|
||||||
u_int _pad; /* make _u 64 bit aligned */
|
u_int _pad; /* make _u 64 bit aligned */
|
||||||
@ -490,6 +491,7 @@ int res_getservers(res_state,
|
|||||||
union res_sockaddr_union *, int);
|
union res_sockaddr_union *, int);
|
||||||
|
|
||||||
void res_setiface();
|
void res_setiface();
|
||||||
|
void res_setmark();
|
||||||
u_int res_randomid(void);
|
u_int res_randomid(void);
|
||||||
|
|
||||||
__END_DECLS
|
__END_DECLS
|
||||||
|
@ -203,6 +203,84 @@ TEST(properties, find_nth) {
|
|||||||
ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(247));
|
ASSERT_EQ((const prop_info *)NULL, __system_property_find_nth(247));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void hierarchical_test_callback(const prop_info *pi, void *cookie) {
|
||||||
|
bool (*ok)[8][8] = static_cast<bool (*)[8][8]>(cookie);
|
||||||
|
|
||||||
|
char name[PROP_NAME_MAX];
|
||||||
|
char value[PROP_VALUE_MAX];
|
||||||
|
|
||||||
|
__system_property_read(pi, name, value);
|
||||||
|
|
||||||
|
int name_i, name_j, name_k;
|
||||||
|
int value_i, value_j, value_k;
|
||||||
|
ASSERT_EQ(3, sscanf(name, "property_%d.%d.%d", &name_i, &name_j, &name_k));
|
||||||
|
ASSERT_EQ(3, sscanf(value, "value_%d.%d.%d", &value_i, &value_j, &value_k));
|
||||||
|
ASSERT_EQ(name_i, value_i);
|
||||||
|
ASSERT_GE(name_i, 0);
|
||||||
|
ASSERT_LT(name_i, 8);
|
||||||
|
ASSERT_EQ(name_j, value_j);
|
||||||
|
ASSERT_GE(name_j, 0);
|
||||||
|
ASSERT_LT(name_j, 8);
|
||||||
|
ASSERT_EQ(name_k, value_k);
|
||||||
|
ASSERT_GE(name_k, 0);
|
||||||
|
ASSERT_LT(name_k, 8);
|
||||||
|
|
||||||
|
ok[name_i][name_j][name_k] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(properties, fill_hierarchical) {
|
||||||
|
LocalPropertyTestState pa;
|
||||||
|
ASSERT_TRUE(pa.valid);
|
||||||
|
char prop_name[PROP_NAME_MAX];
|
||||||
|
char prop_value[PROP_VALUE_MAX];
|
||||||
|
char prop_value_ret[PROP_VALUE_MAX];
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
for (int j = 0; j < 8; j++) {
|
||||||
|
for (int k = 0; k < 8; k++) {
|
||||||
|
ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
|
||||||
|
memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
|
||||||
|
ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
|
||||||
|
memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
|
||||||
|
prop_name[PROP_NAME_MAX - 1] = 0;
|
||||||
|
prop_value[PROP_VALUE_MAX - 1] = 0;
|
||||||
|
|
||||||
|
ASSERT_EQ(0, __system_property_add(prop_name, PROP_NAME_MAX - 1, prop_value, PROP_VALUE_MAX - 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
for (int j = 0; j < 8; j++) {
|
||||||
|
for (int k = 0; k < 8; k++) {
|
||||||
|
ret = snprintf(prop_name, PROP_NAME_MAX - 1, "property_%d.%d.%d", i, j, k);
|
||||||
|
memset(prop_name + ret, 'a', PROP_NAME_MAX - 1 - ret);
|
||||||
|
ret = snprintf(prop_value, PROP_VALUE_MAX - 1, "value_%d.%d.%d", i, j, k);
|
||||||
|
memset(prop_value + ret, 'b', PROP_VALUE_MAX - 1 - ret);
|
||||||
|
prop_name[PROP_NAME_MAX - 1] = 0;
|
||||||
|
prop_value[PROP_VALUE_MAX - 1] = 0;
|
||||||
|
memset(prop_value_ret, '\0', PROP_VALUE_MAX);
|
||||||
|
|
||||||
|
ASSERT_EQ(PROP_VALUE_MAX - 1, __system_property_get(prop_name, prop_value_ret));
|
||||||
|
ASSERT_EQ(0, memcmp(prop_value, prop_value_ret, PROP_VALUE_MAX));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ok[8][8][8];
|
||||||
|
memset(ok, 0, sizeof(ok));
|
||||||
|
__system_property_foreach(hierarchical_test_callback, ok);
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
for (int j = 0; j < 8; j++) {
|
||||||
|
for (int k = 0; k < 8; k++) {
|
||||||
|
ASSERT_TRUE(ok[i][j][k]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
TEST(properties, errors) {
|
TEST(properties, errors) {
|
||||||
LocalPropertyTestState pa;
|
LocalPropertyTestState pa;
|
||||||
ASSERT_TRUE(pa.valid);
|
ASSERT_TRUE(pa.valid);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user