blob: cdb3c80f14e9c63b643edbc3d5b367b191d6e991 [file] [log] [blame]
Nick Kralevich0a230152012-06-04 15:20:25 -07001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#include <string.h>
30#include <stdlib.h>
Elliott Hugheseb847bc2013-10-09 15:50:50 -070031#include "private/libc_logging.h"
Nick Kralevich0a230152012-06-04 15:20:25 -070032
33/*
34 * Runtime implementation of __builtin____strncat_chk.
35 *
36 * See
37 * http://gcc.gnu.org/onlinedocs/gcc/Object-Size-Checking.html
38 * http://gcc.gnu.org/ml/gcc-patches/2004-09/msg02055.html
39 * for details.
40 *
41 * This strncat check is called if _FORTIFY_SOURCE is defined and
42 * greater than 0.
43 */
Nick Kralevichcf870192013-05-30 16:48:53 -070044extern "C" char *__strncat_chk(
45 char* __restrict dest,
46 const char* __restrict src,
47 size_t len, size_t dest_buf_size)
Nick Kralevich0a230152012-06-04 15:20:25 -070048{
Nick Kralevichcf870192013-05-30 16:48:53 -070049 if (len == 0) {
50 return dest;
Nick Kralevich0a230152012-06-04 15:20:25 -070051 }
52
Nick Kralevichcf870192013-05-30 16:48:53 -070053 size_t dest_len = __strlen_chk(dest, dest_buf_size);
54 char *d = dest + dest_len;
55 dest_buf_size -= dest_len;
56
57 while (*src != '\0') {
58 *d++ = *src++;
59 len--; dest_buf_size--;
60
61 if (__predict_false(dest_buf_size == 0)) {
Nick Kralevich6861c6f2013-10-04 09:45:24 -070062 __fortify_chk_fail("strncat prevented write past end of buffer",
Nick Kralevichcf870192013-05-30 16:48:53 -070063 BIONIC_EVENT_STRNCAT_BUFFER_OVERFLOW);
64 }
65
66 if (len == 0) {
67 break;
68 }
Geremy Condra009f3842012-06-08 14:39:11 -070069 }
Nick Kralevich76656af2012-06-07 16:30:02 -070070
Nick Kralevichcf870192013-05-30 16:48:53 -070071 *d = '\0';
72 return dest;
Nick Kralevich0a230152012-06-04 15:20:25 -070073}