13d4c8bbcSPedro F. Giffuni /* $OpenBSD: strlcat.c,v 1.15 2015/03/02 21:41:08 millert Exp $ */
2a41df9e3SWarner Losh
3a41df9e3SWarner Losh /*
43d4c8bbcSPedro F. Giffuni * Copyright (c) 1998, 2015 Todd C. Miller <Todd.Miller@courtesan.com>
5a41df9e3SWarner Losh *
6ba840c78SXin LI * Permission to use, copy, modify, and distribute this software for any
7ba840c78SXin LI * purpose with or without fee is hereby granted, provided that the above
8ba840c78SXin LI * copyright notice and this permission notice appear in all copies.
9a41df9e3SWarner Losh *
10ba840c78SXin LI * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11ba840c78SXin LI * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12ba840c78SXin LI * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13ba840c78SXin LI * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14ba840c78SXin LI * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15ba840c78SXin LI * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16ba840c78SXin LI * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17a41df9e3SWarner Losh */
18a41df9e3SWarner Losh
19a41df9e3SWarner Losh #include <sys/types.h>
20a41df9e3SWarner Losh #include <string.h>
21a41df9e3SWarner Losh
22*cf8e5289SKyle Evans #undef strlcat /* FORTIFY_SOURCE */
23*cf8e5289SKyle Evans
24a41df9e3SWarner Losh /*
253d4c8bbcSPedro F. Giffuni * Appends src to string dst of size dsize (unlike strncat, dsize is the
263d4c8bbcSPedro F. Giffuni * full size of dst, not space left). At most dsize-1 characters
273d4c8bbcSPedro F. Giffuni * will be copied. Always NUL terminates (unless dsize <= strlen(dst)).
283d4c8bbcSPedro F. Giffuni * Returns strlen(src) + MIN(dsize, strlen(initial dst)).
293d4c8bbcSPedro F. Giffuni * If retval >= dsize, truncation occurred.
30a41df9e3SWarner Losh */
316979f76fSKris Kennaway size_t
strlcat(char * __restrict dst,const char * __restrict src,size_t dsize)323d4c8bbcSPedro F. Giffuni strlcat(char * __restrict dst, const char * __restrict src, size_t dsize)
33a41df9e3SWarner Losh {
343d4c8bbcSPedro F. Giffuni const char *odst = dst;
353d4c8bbcSPedro F. Giffuni const char *osrc = src;
363d4c8bbcSPedro F. Giffuni size_t n = dsize;
37a41df9e3SWarner Losh size_t dlen;
38a41df9e3SWarner Losh
393d4c8bbcSPedro F. Giffuni /* Find the end of dst and adjust bytes left but don't go past end. */
403d4c8bbcSPedro F. Giffuni while (n-- != 0 && *dst != '\0')
413d4c8bbcSPedro F. Giffuni dst++;
423d4c8bbcSPedro F. Giffuni dlen = dst - odst;
433d4c8bbcSPedro F. Giffuni n = dsize - dlen;
44a41df9e3SWarner Losh
453d4c8bbcSPedro F. Giffuni if (n-- == 0)
463d4c8bbcSPedro F. Giffuni return(dlen + strlen(src));
473d4c8bbcSPedro F. Giffuni while (*src != '\0') {
483d4c8bbcSPedro F. Giffuni if (n != 0) {
493d4c8bbcSPedro F. Giffuni *dst++ = *src;
50a41df9e3SWarner Losh n--;
51a41df9e3SWarner Losh }
523d4c8bbcSPedro F. Giffuni src++;
53a41df9e3SWarner Losh }
543d4c8bbcSPedro F. Giffuni *dst = '\0';
55a41df9e3SWarner Losh
563d4c8bbcSPedro F. Giffuni return(dlen + (src - osrc)); /* count does not include NUL */
57a41df9e3SWarner Losh }
58