diff options
| author | Elliott Hughes <enh@google.com> | 2015-02-02 17:51:50 +0000 |
|---|---|---|
| committer | Gerrit Code Review <noreply-gerritcodereview@google.com> | 2015-02-02 17:51:50 +0000 |
| commit | 5940f98a35d64f9ca249536bb8cdcdbfe43e9bc3 (patch) | |
| tree | b4dadb078646cbb46dd62a14c623b7e0d06619e3 /libc | |
| parent | ffbda83b302e3c3970135013d2bb92402e040fe5 (diff) | |
| parent | 4674e3899afcc6b3ac8a48cdb716695d5489d26b (diff) | |
| download | bionic-5940f98a35d64f9ca249536bb8cdcdbfe43e9bc3.zip bionic-5940f98a35d64f9ca249536bb8cdcdbfe43e9bc3.tar.gz bionic-5940f98a35d64f9ca249536bb8cdcdbfe43e9bc3.tar.bz2 | |
Merge "Fortify poll and ppoll."
Diffstat (limited to 'libc')
| -rw-r--r-- | libc/Android.mk | 1 | ||||
| -rw-r--r-- | libc/bionic/__poll_chk.cpp | 49 | ||||
| -rw-r--r-- | libc/include/poll.h | 48 |
3 files changed, 96 insertions, 2 deletions
diff --git a/libc/Android.mk b/libc/Android.mk index 691017a..90f1a12 100644 --- a/libc/Android.mk +++ b/libc/Android.mk @@ -69,6 +69,7 @@ libc_common_src_files += \ bionic/__FD_chk.cpp \ bionic/__fgets_chk.cpp \ bionic/__memmove_chk.cpp \ + bionic/__poll_chk.cpp \ bionic/__read_chk.cpp \ bionic/__recvfrom_chk.cpp \ bionic/__stpcpy_chk.cpp \ diff --git a/libc/bionic/__poll_chk.cpp b/libc/bionic/__poll_chk.cpp new file mode 100644 index 0000000..3acac4e --- /dev/null +++ b/libc/bionic/__poll_chk.cpp @@ -0,0 +1,49 @@ +/* + * Copyright (C) 2015 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. + */ + +#undef _FORTIFY_SOURCE +#include <poll.h> +#include "private/libc_logging.h" + +#include <stdio.h> + +extern "C" int __poll_chk(struct pollfd* fds, nfds_t fd_count, int timeout, size_t fds_size) { +fprintf(stderr, "__poll_chk %p %i %i %i\n", fds, (int)fd_count, timeout, (int) fds_size); + if (__predict_false(fds_size / sizeof(*fds) < fd_count)) { + __fortify_chk_fail("poll: pollfd array smaller than fd count", 0); + } + return poll(fds, fd_count, timeout); +} + +extern "C" int __ppoll_chk(struct pollfd* fds, nfds_t fd_count, const struct timespec* timeout, const sigset_t* mask, size_t fds_size) { +fprintf(stderr, "__ppoll_chk %p %i %p %p %i\n", fds, (int)fd_count, timeout, mask, (int) fds_size); + if (__predict_false(fds_size / sizeof(*fds) < fd_count)) { + __fortify_chk_fail("ppoll: pollfd array smaller than fd count", 0); + } + return ppoll(fds, fd_count, timeout, mask); +} diff --git a/libc/include/poll.h b/libc/include/poll.h index 0199cab..7c16d81 100644 --- a/libc/include/poll.h +++ b/libc/include/poll.h @@ -38,8 +38,52 @@ __BEGIN_DECLS typedef unsigned int nfds_t; -extern int poll(struct pollfd*, nfds_t, int); -extern int ppoll(struct pollfd*, nfds_t, const struct timespec*, const sigset_t*); +int poll(struct pollfd*, nfds_t, int); +int ppoll(struct pollfd*, nfds_t, const struct timespec*, const sigset_t*); + +int __poll_chk(struct pollfd*, nfds_t, int, size_t); +int __poll_real(struct pollfd*, nfds_t, int) __RENAME(poll); +__errordecl(__poll_too_small_error, "poll: pollfd array smaller than fd count"); + +int __ppoll_chk(struct pollfd*, nfds_t, const struct timespec*, const sigset_t*, size_t); +int __ppoll_real(struct pollfd*, nfds_t, const struct timespec*, const sigset_t*) __RENAME(ppoll); +__errordecl(__ppoll_too_small_error, "ppoll: pollfd array smaller than fd count"); + +#if defined(__BIONIC_FORTIFY) + +__BIONIC_FORTIFY_INLINE +int poll(struct pollfd* fds, nfds_t fd_count, int timeout) { +#if defined(__clang__) + return __poll_chk(fds, fd_count, timeout, __bos(fds)); +#else + if (__bos(fds) != __BIONIC_FORTIFY_UNKNOWN_SIZE) { + if (!__builtin_constant_p(fd_count)) { + return __poll_chk(fds, fd_count, timeout, __bos(fds)); + } else if (__bos(fds) / sizeof(*fds) < fd_count) { + __poll_too_small_error(); + } + } + return __poll_real(fds, fd_count, timeout); +#endif +} + +__BIONIC_FORTIFY_INLINE +int ppoll(struct pollfd* fds, nfds_t fd_count, const struct timespec* timeout, const sigset_t* mask) { +#if defined(__clang__) + return __ppoll_chk(fds, fd_count, timeout, mask, __bos(fds)); +#else + if (__bos(fds) != __BIONIC_FORTIFY_UNKNOWN_SIZE) { + if (!__builtin_constant_p(fd_count)) { + return __ppoll_chk(fds, fd_count, timeout, mask, __bos(fds)); + } else if (__bos(fds) / sizeof(*fds) < fd_count) { + __ppoll_too_small_error(); + } + } + return __ppoll_real(fds, fd_count, timeout, mask); +#endif +} + +#endif __END_DECLS |
