File: strlen.c

package info (click to toggle)
picolibc 1.8.11-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 50,064 kB
  • sloc: ansic: 404,031; asm: 24,984; sh: 2,585; python: 2,289; perl: 680; pascal: 329; exp: 287; makefile: 222; cpp: 71; xml: 40
file content (80 lines) | stat: -rw-r--r-- 2,153 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
/*
Copyright (c) 1994 Cygnus Support.
All rights reserved.

Redistribution and use in source and binary forms are permitted
provided that the above copyright notice and this paragraph are
duplicated in all such forms and that any documentation,
and/or other materials related to such
distribution and use acknowledge that the software was developed
at Cygnus Support, Inc.  Cygnus Support, Inc. may not be used to
endorse or promote products derived from this software without
specific prior written permission.
THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
 */
/*
FUNCTION
        <<strlen>>---character string length

INDEX
        strlen

SYNOPSIS
        #include <string.h>
        size_t strlen(const char *<[str]>);

DESCRIPTION
        The <<strlen>> function works out the length of the string
        starting at <<*<[str]>>> by counting chararacters until it
        reaches a <<NULL>> character.

RETURNS
        <<strlen>> returns the character count.

PORTABILITY
<<strlen>> is ANSI C.

<<strlen>> requires no supporting OS subroutines.

QUICKREF
        strlen ansi pure
*/

#include <string.h>
#include <limits.h>
#include "local.h"

size_t
strlen(const char *str)
{
    const char *start = str;

#if !defined(__PREFER_SIZE_OVER_SPEED) && !defined(__OPTIMIZE_SIZE__) \
    && !defined(_PICOLIBC_NO_OUT_OF_BOUNDS_READS)
    unsigned long *aligned_addr;

    /* Align the pointer, so we can search a word at a time.  */
    while (UNALIGNED_X(str)) {
        if (!*str)
            return str - start;
        str++;
    }

    /* If the string is word-aligned, we can check for the presence of
       a null in each word-sized block.  */
    aligned_addr = (unsigned long *)str;
    while (!DETECT_NULL(*aligned_addr))
        aligned_addr++;

    /* Once a null is detected, we check each byte in that block for a
       precise position of the null.  */
    str = (char *)aligned_addr;

#endif /* not __PREFER_SIZE_OVER_SPEED */

    while (*str)
        str++;
    return str - start;
}